mirror of
https://github.com/baldurk/renderdoc.git
synced 2026-05-05 01:20:42 +00:00
611 lines
19 KiB
C++
611 lines
19 KiB
C++
/******************************************************************************
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2019-2025 Baldur Karlsson
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
******************************************************************************/
|
|
|
|
#include "CameraControlsDialog.h"
|
|
#include <QKeyEvent>
|
|
#include <QMetaEnum>
|
|
#include <QMouseEvent>
|
|
#include "Code/QRDUtils.h"
|
|
#include "ui_CameraControlsDialog.h"
|
|
|
|
CameraControlsDialog::CameraControlsDialog(ICaptureContext &Ctx, QWidget *parent)
|
|
: QDialog(parent), m_Ctx(Ctx), ui(new Ui::CameraControlsDialog)
|
|
{
|
|
ui->setupUi(this);
|
|
setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
|
|
ui->nearPlane->setValue(m_Ctx.Config().MeshViewer_CameraNear);
|
|
ui->farPlane->setValue(m_Ctx.Config().MeshViewer_CameraFar);
|
|
|
|
m_Keys = m_Ctx.Config().MeshViewer_KeySettings;
|
|
|
|
updateDisplayLabels();
|
|
|
|
ui->speedMod->setCurrentIndex(0);
|
|
if(m_Ctx.Config().MeshViewer_SpeedModifier == Qt::ShiftModifier)
|
|
ui->speedMod->setCurrentIndex(0);
|
|
else if(m_Ctx.Config().MeshViewer_SpeedModifier == Qt::AltModifier)
|
|
ui->speedMod->setCurrentIndex(1);
|
|
else if(m_Ctx.Config().MeshViewer_SpeedModifier == Qt::ControlModifier)
|
|
ui->speedMod->setCurrentIndex(2);
|
|
else if(m_Ctx.Config().MeshViewer_SpeedModifier == Qt::NoModifier)
|
|
ui->speedMod->setCurrentIndex(3);
|
|
|
|
m_Keys.resize((size_t)KeyPressDirection::NumSettings);
|
|
|
|
for(QToolButton *b : findChildren<QToolButton *>())
|
|
connect(b, &QToolButton::clicked, this, &CameraControlsDialog::setKey);
|
|
|
|
connect(ui->buttonBox, &QDialogButtonBox::accepted, this,
|
|
&CameraControlsDialog::applyUpdatedControls);
|
|
connect(ui->buttonBox, &QDialogButtonBox::rejected, this, &QDialog::reject);
|
|
|
|
{
|
|
m_KeybindDialog = new QDialog(this);
|
|
m_KeybindDialog->setWindowTitle(tr("Make Key bind"));
|
|
m_KeybindDialog->setWindowFlags(m_KeybindDialog->windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
m_KeybindDialog->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
|
|
m_KeybindDialog->setFixedSize(200, 80);
|
|
|
|
QDialogButtonBox *buttons = new QDialogButtonBox(m_KeybindDialog);
|
|
buttons->addButton(QDialogButtonBox::Cancel);
|
|
QObject::connect(buttons, &QDialogButtonBox::rejected, m_KeybindDialog, &QDialog::reject);
|
|
|
|
QLabel *instructions = new QLabel(m_KeybindDialog);
|
|
instructions->setText(tr("Press key or mouse button, or escape to unbind."));
|
|
instructions->setWordWrap(true);
|
|
|
|
QVBoxLayout *layout = new QVBoxLayout(m_KeybindDialog);
|
|
|
|
layout->addWidget(instructions);
|
|
layout->addWidget(buttons);
|
|
|
|
m_KeybindDialog->setLayout(layout);
|
|
|
|
m_KeybindDialog->installEventFilter(this);
|
|
}
|
|
}
|
|
|
|
void CameraControlsDialog::applyUpdatedControls()
|
|
{
|
|
m_Ctx.Config().MeshViewer_CameraNear = (float)ui->nearPlane->value();
|
|
m_Ctx.Config().MeshViewer_CameraFar = (float)ui->farPlane->value();
|
|
|
|
switch(ui->speedMod->currentIndex())
|
|
{
|
|
case 0: m_Ctx.Config().MeshViewer_SpeedModifier = Qt::ShiftModifier; break;
|
|
case 1: m_Ctx.Config().MeshViewer_SpeedModifier = Qt::AltModifier; break;
|
|
case 2: m_Ctx.Config().MeshViewer_SpeedModifier = Qt::ControlModifier; break;
|
|
default: m_Ctx.Config().MeshViewer_SpeedModifier = Qt::NoModifier; break;
|
|
}
|
|
|
|
m_Ctx.Config().MeshViewer_KeySettings = m_Keys;
|
|
|
|
m_Ctx.Config().Save();
|
|
accept();
|
|
}
|
|
|
|
void CameraControlsDialog::on_resetAll_clicked()
|
|
{
|
|
m_Keys.clear();
|
|
m_Keys.resize((size_t)KeyPressDirection::NumSettings);
|
|
|
|
updateDisplayLabels();
|
|
}
|
|
|
|
void CameraControlsDialog::setKey()
|
|
{
|
|
m_Keybind = 0;
|
|
RDDialog::show(m_KeybindDialog);
|
|
|
|
// find the corresponding display label for this button
|
|
QLineEdit *display =
|
|
findChild<QLineEdit *>(QObject::sender()->objectName().replace(lit("Set"), lit("Display")));
|
|
|
|
const QLineEdit *labels[(size_t)KeyPressDirection::NumSettings] = {
|
|
// KeyPressDirection::Forward,
|
|
ui->forwardDisplay,
|
|
ui->forwardDisplay_2,
|
|
// KeyPressDirection::Back,
|
|
ui->backwardDisplay,
|
|
ui->backwardDisplay_2,
|
|
// KeyPressDirection::Left,
|
|
ui->leftDisplay,
|
|
ui->leftDisplay_2,
|
|
// KeyPressDirection::Right,
|
|
ui->rightDisplay,
|
|
ui->rightDisplay_2,
|
|
// KeyPressDirection::Up,
|
|
ui->upDisplay,
|
|
ui->upDisplay_2,
|
|
// KeyPressDirection::Down,
|
|
ui->downDisplay,
|
|
ui->downDisplay_2,
|
|
};
|
|
|
|
int keyIdx = -1;
|
|
for(int i = 0; i < (int)ARRAY_COUNT(labels); i++)
|
|
{
|
|
if(labels[i] == display)
|
|
{
|
|
keyIdx = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(keyIdx < 0)
|
|
{
|
|
qCritical() << "Couldn't identify key being bound";
|
|
return;
|
|
}
|
|
|
|
if(m_Keybind == 0)
|
|
{
|
|
// cancelled, do nothing
|
|
return;
|
|
}
|
|
else if(getKeySetting(m_Keybind) == Qt::Key_Escape)
|
|
{
|
|
m_Keys[keyIdx] = makeUnboundSetting();
|
|
}
|
|
else
|
|
{
|
|
bool update = true;
|
|
|
|
// check for duplicates
|
|
for(size_t i = 0; i < m_Keys.size(); i++)
|
|
{
|
|
const KeyPressDirection dir = KeyPressDirection(i / 2);
|
|
const bool isPrimary = (i % 2) == 0;
|
|
|
|
bool isDefault = false;
|
|
uint32_t k = m_Keys[i];
|
|
if(k == 0)
|
|
{
|
|
k = getDefaultKey(dir, isPrimary);
|
|
isDefault = true;
|
|
}
|
|
|
|
if(k == m_Keybind && keyIdx != (int)i)
|
|
{
|
|
QString idxName;
|
|
switch(dir)
|
|
{
|
|
case KeyPressDirection::Left: idxName = tr("Left"); break;
|
|
case KeyPressDirection::Right: idxName = tr("Right"); break;
|
|
case KeyPressDirection::Forward: idxName = tr("Forward"); break;
|
|
case KeyPressDirection::Back: idxName = tr("Back"); break;
|
|
case KeyPressDirection::Up: idxName = tr("Up"); break;
|
|
case KeyPressDirection::Down: idxName = tr("Down"); break;
|
|
default: idxName = lit("???"); break;
|
|
}
|
|
|
|
if(isPrimary)
|
|
idxName += tr(" - Primary");
|
|
else
|
|
idxName += tr(" - Secondary");
|
|
|
|
if(isDefault)
|
|
idxName += tr(" (Default bind)");
|
|
|
|
QMessageBox::StandardButton ret =
|
|
RDDialog::question(this, tr("Conflicting keybind"),
|
|
tr("%1 is already bound to %2. Continue and unbind old key?")
|
|
.arg(nameForSetting(m_Keybind))
|
|
.arg(idxName));
|
|
|
|
if(ret == QMessageBox::No)
|
|
update = false;
|
|
else if(ret == QMessageBox::Yes)
|
|
m_Keys[i] = makeUnboundSetting();
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(update)
|
|
m_Keys[keyIdx] = m_Keybind;
|
|
}
|
|
|
|
updateDisplayLabels();
|
|
}
|
|
|
|
bool CameraControlsDialog::eventFilter(QObject *watched, QEvent *event)
|
|
{
|
|
if(event->type() == QEvent::KeyPress || event->type() == QEvent::ShortcutOverride)
|
|
{
|
|
QKeyEvent *key = (QKeyEvent *)event;
|
|
|
|
if(key->key() != 0 && key->key() != Qt::Key_unknown && key->key() != Qt::Key_Alt &&
|
|
key->key() != Qt::Key_AltGr && key->key() != Qt::Key_Control && key->key() != Qt::Key_Meta &&
|
|
key->key() != Qt::Key_Shift && key->key() != Qt::Key_CapsLock &&
|
|
key->key() != Qt::Key_NumLock && key->key() != Qt::Key_ScrollLock &&
|
|
key->key() != Qt::Key_Kana_Lock && key->key() != Qt::Key_Kana_Shift &&
|
|
key->key() != Qt::Key_Eisu_Shift && key->key() != Qt::Key_Eisu_toggle)
|
|
{
|
|
m_Keybind = makeKeySetting((Qt::Key)key->key());
|
|
m_KeybindDialog->accept();
|
|
event->accept();
|
|
return true;
|
|
}
|
|
}
|
|
else if(event->type() == QEvent::MouseButtonPress)
|
|
{
|
|
QMouseEvent *mouse = (QMouseEvent *)event;
|
|
if(mouse->button() != Qt::LeftButton && mouse->button() != Qt::RightButton &&
|
|
mouse->button() != Qt::MiddleButton)
|
|
{
|
|
m_Keybind = makeMouseButtonSetting(mouse->button());
|
|
m_KeybindDialog->accept();
|
|
event->accept();
|
|
return true;
|
|
}
|
|
}
|
|
else if(event->type() == QEvent::Wheel)
|
|
{
|
|
QWheelEvent *mouse = (QWheelEvent *)event;
|
|
|
|
m_Keybind = makeMouseWheelSetting(mouse->angleDelta());
|
|
m_KeybindDialog->accept();
|
|
event->accept();
|
|
return true;
|
|
}
|
|
|
|
return QObject::eventFilter(watched, event);
|
|
}
|
|
|
|
QString CameraControlsDialog::buttonName(Qt::MouseButton button)
|
|
{
|
|
switch(button)
|
|
{
|
|
case Qt::NoButton: return tr("No mouse button");
|
|
case Qt::LeftButton: return tr("Left mouse");
|
|
case Qt::MiddleButton: return tr("Middle mouse");
|
|
case Qt::RightButton: return tr("Right mouse");
|
|
case Qt::BackButton: return tr("Mouse back");
|
|
case Qt::ForwardButton: return tr("Mouse forward");
|
|
case Qt::TaskButton: return tr("Mouse task");
|
|
case Qt::ExtraButton4: return tr("Mouse 7");
|
|
case Qt::ExtraButton5: return tr("Mouse 8");
|
|
case Qt::ExtraButton6: return tr("Mouse 9");
|
|
case Qt::ExtraButton7: return tr("Mouse 10");
|
|
case Qt::ExtraButton8: return tr("Mouse 11");
|
|
case Qt::ExtraButton9: return tr("Mouse 12");
|
|
case Qt::ExtraButton10: return tr("Mouse 13");
|
|
case Qt::ExtraButton11: return tr("Mouse 14");
|
|
case Qt::ExtraButton12: return tr("Mouse 15");
|
|
case Qt::ExtraButton13: return tr("Mouse 16");
|
|
case Qt::ExtraButton14: return tr("Mouse 17");
|
|
case Qt::ExtraButton15: return tr("Mouse 18");
|
|
case Qt::ExtraButton16: return tr("Mouse 19");
|
|
case Qt::ExtraButton17: return tr("Mouse 20");
|
|
case Qt::ExtraButton18: return tr("Mouse 21");
|
|
case Qt::ExtraButton19: return tr("Mouse 22");
|
|
case Qt::ExtraButton20: return tr("Mouse 23");
|
|
case Qt::ExtraButton21: return tr("Mouse 24");
|
|
case Qt::ExtraButton22: return tr("Mouse 25");
|
|
case Qt::ExtraButton23: return tr("Mouse 26");
|
|
case Qt::ExtraButton24: return tr("Mouse 27");
|
|
default: return tr("Unknown button");
|
|
}
|
|
}
|
|
|
|
QString CameraControlsDialog::wheelName(QPoint angleDelta)
|
|
{
|
|
if(angleDelta.y() > 0)
|
|
return tr("Mousewheel up");
|
|
else if(angleDelta.y() < 0)
|
|
return tr("Mousewheel down");
|
|
else if(angleDelta.x() < 0)
|
|
return tr("Mousewheel left");
|
|
else if(angleDelta.x() > 0)
|
|
return tr("Mousewheel right");
|
|
return tr("Unknown wheel");
|
|
}
|
|
|
|
QString CameraControlsDialog::nameForSetting(uint32_t k)
|
|
{
|
|
if(getKeySetting(k) != Qt::Key_unknown)
|
|
return QKeySequence(getKeySetting(k)).toString();
|
|
else if(getMouseButtonSetting(k) != Qt::MaxMouseButton)
|
|
return buttonName(getMouseButtonSetting(k));
|
|
else if(getMouseWheelSetting(k) != QPoint())
|
|
return wheelName(getMouseWheelSetting(k));
|
|
|
|
return tr("Unbound");
|
|
}
|
|
|
|
void CameraControlsDialog::updateDisplayLabels()
|
|
{
|
|
QLineEdit *labels[(size_t)KeyPressDirection::NumSettings] = {
|
|
// KeyPressDirection::Forward,
|
|
ui->forwardDisplay,
|
|
ui->forwardDisplay_2,
|
|
// KeyPressDirection::Back,
|
|
ui->backwardDisplay,
|
|
ui->backwardDisplay_2,
|
|
// KeyPressDirection::Left,
|
|
ui->leftDisplay,
|
|
ui->leftDisplay_2,
|
|
// KeyPressDirection::Right,
|
|
ui->rightDisplay,
|
|
ui->rightDisplay_2,
|
|
// KeyPressDirection::Up,
|
|
ui->upDisplay,
|
|
ui->upDisplay_2,
|
|
// KeyPressDirection::Down,
|
|
ui->downDisplay,
|
|
ui->downDisplay_2,
|
|
};
|
|
|
|
for(size_t i = 0; i < (size_t)KeyPressDirection::Count; i++)
|
|
{
|
|
for(size_t j = 0; j < 2; j++)
|
|
{
|
|
size_t idx = i * 2 + j;
|
|
|
|
if(idx >= m_Keys.size() || m_Keys[idx] == 0)
|
|
{
|
|
labels[idx]->setText(
|
|
tr("Default - %1")
|
|
.arg(nameForSetting(makeKeySetting(getDefaultKey(KeyPressDirection(i), j == 0)))));
|
|
}
|
|
else
|
|
{
|
|
labels[idx]->setText(nameForSetting(m_Keys[idx]));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
CameraControlsDialog::~CameraControlsDialog()
|
|
{
|
|
delete ui;
|
|
}
|
|
|
|
namespace NativeScanCode
|
|
{
|
|
enum
|
|
{
|
|
#if defined(Q_OS_WIN32)
|
|
Key_A = 30,
|
|
Key_S = 31,
|
|
Key_D = 32,
|
|
Key_F = 33,
|
|
Key_W = 17,
|
|
Key_R = 19,
|
|
#elif defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
|
|
Key_A = 30 + 8,
|
|
Key_S = 31 + 8,
|
|
Key_D = 32 + 8,
|
|
Key_F = 33 + 8,
|
|
Key_W = 17 + 8,
|
|
Key_R = 19 + 8,
|
|
#elif defined(Q_OS_MACOS)
|
|
// scan codes not supported on OS X
|
|
Key_A = 0xDEADBEF1,
|
|
Key_S = 0xDEADBEF2,
|
|
Key_D = 0xDEADBEF3,
|
|
Key_F = 0xDEADBEF4,
|
|
Key_W = 0xDEADBEF5,
|
|
Key_R = 0xDEADBEF6,
|
|
#else
|
|
#error "Unknown platform! Define NativeScanCode"
|
|
#endif
|
|
};
|
|
}; // namespace NativeScanCode
|
|
|
|
// default to wasd, this will not work with other keyboard layouts but is the only fallback we have
|
|
static Qt::Key defaultPrimaryKeys[(size_t)KeyPressDirection::Count] = {
|
|
Qt::Key_W, Qt::Key_S, Qt::Key_A, Qt::Key_D, Qt::Key_R, Qt::Key_F,
|
|
};
|
|
// the default secondaries are fortunately fixed keys
|
|
static const Qt::Key defaultSecondaryKeys[(size_t)KeyPressDirection::Count] = {
|
|
Qt::Key_Up, Qt::Key_Down, Qt::Key_Left, Qt::Key_Right, Qt::Key_PageUp, Qt::Key_PageDown,
|
|
};
|
|
static bool primaryKeysFilled = false;
|
|
|
|
#if defined(Q_OS_WIN32)
|
|
#define NOMINMAX
|
|
#include <windows.h>
|
|
|
|
void FetchDefaultPrimaryKeys()
|
|
{
|
|
if(primaryKeysFilled)
|
|
return;
|
|
|
|
primaryKeysFilled = true;
|
|
|
|
static int scans[(size_t)KeyPressDirection::Count] = {
|
|
NativeScanCode::Key_W, NativeScanCode::Key_S, NativeScanCode::Key_A,
|
|
NativeScanCode::Key_D, NativeScanCode::Key_R, NativeScanCode::Key_F,
|
|
};
|
|
|
|
byte state[256] = {};
|
|
for(size_t i = 0; i < (size_t)KeyPressDirection::Count; i++)
|
|
{
|
|
uint vk = MapVirtualKeyW(scans[i], MAPVK_VSC_TO_VK);
|
|
|
|
wchar_t buf[8] = {};
|
|
int res = ToUnicode(vk, scans[i], state, buf, 7, 0);
|
|
|
|
if(res == 0)
|
|
{
|
|
qCritical() << "couldn't get key for" << i;
|
|
}
|
|
else
|
|
{
|
|
defaultPrimaryKeys[i] = Qt::Key(QChar(buf[0]).toUpper().unicode());
|
|
}
|
|
}
|
|
}
|
|
|
|
#elif defined(Q_OS_LINUX)
|
|
|
|
#include <dlfcn.h>
|
|
#include <QX11Info>
|
|
|
|
// predeclare enough of xkbcommon, so we don't have a new build time dependency on it. Qt will load it for us
|
|
extern "C" {
|
|
struct xkb_state;
|
|
struct xkb_context;
|
|
struct xkb_keymap;
|
|
typedef uint32_t xkb_keysym_t;
|
|
typedef uint32_t xkb_keycode_t;
|
|
enum xkb_context_flags
|
|
{
|
|
XKB_CONTEXT_NO_DEFAULT_INCLUDES = 1,
|
|
};
|
|
enum xkb_keymap_compile_flags
|
|
{
|
|
XKB_KEYMAP_COMPILE_NO_FLAGS = 0,
|
|
};
|
|
|
|
xkb_context *xkb_context_new(xkb_context_flags flags);
|
|
int32_t xkb_x11_get_core_keyboard_device_id(xcb_connection_t *connection);
|
|
xkb_keymap *xkb_x11_keymap_new_from_device(xkb_context *context, xcb_connection_t *connection,
|
|
int32_t device_id, xkb_keymap_compile_flags flags);
|
|
xkb_state *xkb_x11_state_new_from_device(xkb_keymap *keymap, xcb_connection_t *connection,
|
|
int32_t device_id);
|
|
xkb_keysym_t xkb_state_key_get_one_sym(xkb_state *state, xkb_keycode_t key);
|
|
int xkb_keysym_to_utf8(xkb_keysym_t keysym, char *buffer, size_t size);
|
|
void xkb_state_unref(xkb_state *state);
|
|
void xkb_context_unref(xkb_context *context);
|
|
void xkb_keymap_unref(xkb_keymap *keymap);
|
|
|
|
using PFN_xkb_context_new = decltype(&xkb_context_new);
|
|
using PFN_xkb_x11_get_core_keyboard_device_id = decltype(&xkb_x11_get_core_keyboard_device_id);
|
|
using PFN_xkb_x11_keymap_new_from_device = decltype(&xkb_x11_keymap_new_from_device);
|
|
using PFN_xkb_x11_state_new_from_device = decltype(&xkb_x11_state_new_from_device);
|
|
using PFN_xkb_state_key_get_one_sym = decltype(&xkb_state_key_get_one_sym);
|
|
using PFN_xkb_keysym_to_utf8 = decltype(&xkb_keysym_to_utf8);
|
|
using PFN_xkb_state_unref = decltype(&xkb_state_unref);
|
|
using PFN_xkb_context_unref = decltype(&xkb_context_unref);
|
|
using PFN_xkb_keymap_unref = decltype(&xkb_keymap_unref);
|
|
};
|
|
|
|
void *findXKBSym(const char *name)
|
|
{
|
|
static void *searchLibs[] = {
|
|
RTLD_DEFAULT,
|
|
dlopen("libxkbcommon.so", RTLD_NOW | RTLD_LOCAL),
|
|
dlopen("libxkbcommon-x11.so", RTLD_NOW | RTLD_LOCAL),
|
|
dlopen("libxkbcommon-x11.so.0", RTLD_NOW | RTLD_LOCAL),
|
|
};
|
|
|
|
for(void *lib : searchLibs)
|
|
{
|
|
void *ret = dlsym(lib, name);
|
|
if(ret)
|
|
return ret;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void FetchDefaultPrimaryKeys()
|
|
{
|
|
if(primaryKeysFilled)
|
|
return;
|
|
|
|
primaryKeysFilled = true;
|
|
|
|
PFN_xkb_context_new dyn_xkb_context_new = (PFN_xkb_context_new)findXKBSym("xkb_context_new");
|
|
PFN_xkb_x11_get_core_keyboard_device_id dyn_xkb_x11_get_core_keyboard_device_id =
|
|
(PFN_xkb_x11_get_core_keyboard_device_id)findXKBSym("xkb_x11_get_core_keyboard_device_id");
|
|
PFN_xkb_x11_keymap_new_from_device dyn_xkb_x11_keymap_new_from_device =
|
|
(PFN_xkb_x11_keymap_new_from_device)findXKBSym("xkb_x11_keymap_new_from_device");
|
|
PFN_xkb_x11_state_new_from_device dyn_xkb_x11_state_new_from_device =
|
|
(PFN_xkb_x11_state_new_from_device)findXKBSym("xkb_x11_state_new_from_device");
|
|
PFN_xkb_state_key_get_one_sym dyn_xkb_state_key_get_one_sym =
|
|
(PFN_xkb_state_key_get_one_sym)findXKBSym("xkb_state_key_get_one_sym");
|
|
PFN_xkb_keysym_to_utf8 dyn_xkb_keysym_to_utf8 =
|
|
(PFN_xkb_keysym_to_utf8)findXKBSym("xkb_keysym_to_utf8");
|
|
PFN_xkb_state_unref dyn_xkb_state_unref = (PFN_xkb_state_unref)findXKBSym("xkb_state_unref");
|
|
PFN_xkb_context_unref dyn_xkb_context_unref =
|
|
(PFN_xkb_context_unref)findXKBSym("xkb_context_unref");
|
|
PFN_xkb_keymap_unref dyn_xkb_keymap_unref = (PFN_xkb_keymap_unref)findXKBSym("xkb_keymap_unref");
|
|
|
|
// if both general and xcb symbols loaded, we're good to go
|
|
if(dyn_xkb_context_new && dyn_xkb_x11_keymap_new_from_device)
|
|
{
|
|
xcb_connection_t *connection = QX11Info::connection();
|
|
xkb_context *context = dyn_xkb_context_new(XKB_CONTEXT_NO_DEFAULT_INCLUDES);
|
|
int core_device_id = dyn_xkb_x11_get_core_keyboard_device_id(connection);
|
|
xkb_keymap *keymap = dyn_xkb_x11_keymap_new_from_device(context, connection, core_device_id,
|
|
XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
xkb_state *state = dyn_xkb_x11_state_new_from_device(keymap, connection, core_device_id);
|
|
|
|
static int scans[(size_t)KeyPressDirection::Count] = {
|
|
NativeScanCode::Key_W, NativeScanCode::Key_S, NativeScanCode::Key_A,
|
|
NativeScanCode::Key_D, NativeScanCode::Key_R, NativeScanCode::Key_F,
|
|
};
|
|
|
|
for(size_t i = 0; i < (size_t)KeyPressDirection::Count; i++)
|
|
{
|
|
xkb_keysym_t sym = dyn_xkb_state_key_get_one_sym(state, scans[i]);
|
|
|
|
char buf[32] = {};
|
|
int len = dyn_xkb_keysym_to_utf8(sym, buf, 31);
|
|
|
|
if(len == 0)
|
|
{
|
|
qCritical() << "couldn't get key for" << i;
|
|
}
|
|
else
|
|
{
|
|
defaultPrimaryKeys[i] = Qt::Key(QString::fromUtf8(buf).unicode()->toUpper().unicode());
|
|
}
|
|
}
|
|
|
|
dyn_xkb_state_unref(state);
|
|
dyn_xkb_keymap_unref(keymap);
|
|
dyn_xkb_context_unref(context);
|
|
}
|
|
}
|
|
|
|
#else
|
|
|
|
void FetchDefaultPrimaryKeys()
|
|
{
|
|
if(primaryKeysFilled)
|
|
return;
|
|
|
|
primaryKeysFilled = true;
|
|
|
|
qCritical() << "Unsupported platform to fetch default primary keys";
|
|
}
|
|
|
|
#endif
|
|
|
|
Qt::Key getDefaultKey(KeyPressDirection dir, bool primary)
|
|
{
|
|
FetchDefaultPrimaryKeys();
|
|
|
|
if(primary)
|
|
return defaultPrimaryKeys[(size_t)dir];
|
|
|
|
return defaultSecondaryKeys[(size_t)dir];
|
|
}
|