2022-08-05 13:03:37 +02:00
|
|
|
#include "InputMethodRelay.hpp"
|
|
|
|
#include "InputManager.hpp"
|
2022-08-05 16:21:08 +02:00
|
|
|
#include "../../Compositor.hpp"
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
CInputMethodRelay::CInputMethodRelay() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::onNewIME(wlr_input_method_v2* pIME) {
|
|
|
|
if (m_pWLRIME) {
|
|
|
|
Debug::log(ERR, "Cannot register 2 IMEs at once!");
|
|
|
|
|
|
|
|
wlr_input_method_v2_send_unavailable(pIME);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_pWLRIME = pIME;
|
|
|
|
|
|
|
|
hyprListener_IMECommit.initCallback(&m_pWLRIME->events.commit, [&](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PTI = getFocusedTextInput();
|
|
|
|
const auto PIMR = (CInputMethodRelay*)owner;
|
|
|
|
|
|
|
|
if (!PTI) {
|
|
|
|
Debug::log(LOG, "No focused TextInput on IME Commit");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PIMR->m_pWLRIME->current.preedit.text) {
|
|
|
|
wlr_text_input_v3_send_preedit_string(PTI->pWlrInput, PIMR->m_pWLRIME->current.preedit.text, PIMR->m_pWLRIME->current.preedit.cursor_begin, PIMR->m_pWLRIME->current.preedit.cursor_end);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PIMR->m_pWLRIME->current.commit_text) {
|
|
|
|
wlr_text_input_v3_send_commit_string(PTI->pWlrInput, PIMR->m_pWLRIME->current.commit_text);
|
|
|
|
}
|
|
|
|
|
2022-08-05 16:21:08 +02:00
|
|
|
if (PIMR->m_pWLRIME->current.delete_.before_length || PIMR->m_pWLRIME->current.delete_.after_length) {
|
2022-08-05 13:03:37 +02:00
|
|
|
wlr_text_input_v3_send_delete_surrounding_text(PTI->pWlrInput, PIMR->m_pWLRIME->current.delete_.before_length, PIMR->m_pWLRIME->current.delete_.after_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_text_input_v3_send_done(PTI->pWlrInput);
|
|
|
|
|
|
|
|
}, this, "IMERelay");
|
|
|
|
|
|
|
|
hyprListener_IMEDestroy.initCallback(&m_pWLRIME->events.destroy, [&](void* owner, void* data) {
|
|
|
|
|
|
|
|
m_pWLRIME = nullptr;
|
|
|
|
|
|
|
|
hyprListener_IMEDestroy.removeCallback();
|
|
|
|
hyprListener_IMECommit.removeCallback();
|
2022-08-05 16:21:08 +02:00
|
|
|
hyprListener_IMEGrab.removeCallback();
|
2022-08-05 17:07:01 +02:00
|
|
|
hyprListener_IMENewPopup.removeCallback();
|
2022-08-05 16:21:08 +02:00
|
|
|
|
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
const auto PTI = getFocusedTextInput();
|
|
|
|
|
|
|
|
Debug::log(LOG, "IME Destroy");
|
|
|
|
|
|
|
|
if (PTI) {
|
2022-08-05 13:19:16 +02:00
|
|
|
setPendingSurface(PTI, PTI->pWlrInput->focused_surface);
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
wlr_text_input_v3_send_leave(PTI->pWlrInput);
|
|
|
|
}
|
|
|
|
|
|
|
|
}, this, "IMERelay");
|
|
|
|
|
2022-08-05 16:21:08 +02:00
|
|
|
hyprListener_IMEGrab.initCallback(&m_pWLRIME->events.grab_keyboard, [&](void* owner, void* data) {
|
|
|
|
|
|
|
|
Debug::log(LOG, "IME TextInput Keyboard Grab new");
|
|
|
|
|
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
|
|
|
|
|
|
|
m_pKeyboardGrab = std::make_unique<SIMEKbGrab>();
|
|
|
|
|
|
|
|
m_pKeyboardGrab->pKeyboard = wlr_seat_get_keyboard(g_pCompositor->m_sSeat.seat);
|
|
|
|
|
|
|
|
const auto PKBGRAB = (wlr_input_method_keyboard_grab_v2*)data;
|
|
|
|
|
|
|
|
m_pKeyboardGrab->pWlrKbGrab = PKBGRAB;
|
|
|
|
|
|
|
|
wlr_input_method_keyboard_grab_v2_set_keyboard(m_pKeyboardGrab->pWlrKbGrab, m_pKeyboardGrab->pKeyboard);
|
|
|
|
|
|
|
|
m_pKeyboardGrab->hyprListener_grabDestroy.initCallback(&PKBGRAB->events.destroy, [&](void* owner, void* data) {
|
|
|
|
|
|
|
|
m_pKeyboardGrab->hyprListener_grabDestroy.removeCallback();
|
|
|
|
|
|
|
|
Debug::log(LOG, "IME TextInput Keyboard Grab destroy");
|
|
|
|
|
|
|
|
if (m_pKeyboardGrab->pKeyboard) {
|
|
|
|
wlr_seat_keyboard_notify_modifiers(g_pCompositor->m_sSeat.seat, &m_pKeyboardGrab->pKeyboard->modifiers);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
|
|
|
|
|
|
|
}, m_pKeyboardGrab.get(), "IME Keyboard Grab");
|
|
|
|
|
|
|
|
}, this, "IMERelay");
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
hyprListener_IMENewPopup.initCallback(&m_pWLRIME->events.new_popup_surface, [&](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PNEWPOPUP = &m_lIMEPopups.emplace_back();
|
|
|
|
|
|
|
|
PNEWPOPUP->pSurface = (wlr_input_popup_surface_v2*)data;
|
|
|
|
|
|
|
|
PNEWPOPUP->hyprListener_commitPopup.initCallback(&PNEWPOPUP->pSurface->surface->events.commit, &Events::listener_commitInputPopup, PNEWPOPUP, "IME Popup");
|
|
|
|
PNEWPOPUP->hyprListener_mapPopup.initCallback(&PNEWPOPUP->pSurface->events.map, &Events::listener_mapInputPopup, PNEWPOPUP, "IME Popup");
|
|
|
|
PNEWPOPUP->hyprListener_unmapPopup.initCallback(&PNEWPOPUP->pSurface->events.unmap, &Events::listener_unmapInputPopup, PNEWPOPUP, "IME Popup");
|
|
|
|
PNEWPOPUP->hyprListener_destroyPopup.initCallback(&PNEWPOPUP->pSurface->events.destroy, &Events::listener_destroyInputPopup, PNEWPOPUP, "IME Popup");
|
|
|
|
|
|
|
|
Debug::log(LOG, "New input popup");
|
|
|
|
|
|
|
|
}, this, "IMERelay");
|
|
|
|
|
2022-08-05 13:03:37 +02:00
|
|
|
const auto PTI = getFocusableTextInput();
|
|
|
|
|
|
|
|
if (PTI) {
|
|
|
|
wlr_text_input_v3_send_enter(PTI->pWlrInput, PTI->pPendingSurface);
|
2022-08-05 13:19:16 +02:00
|
|
|
setPendingSurface(PTI, nullptr);
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
void CInputMethodRelay::updateInputPopup(SIMEPopup* pPopup) {
|
|
|
|
if (!pPopup->pSurface->mapped)
|
|
|
|
return;
|
|
|
|
|
2022-08-23 11:16:27 +02:00
|
|
|
// damage last known pos & size
|
|
|
|
g_pHyprRenderer->damageBox(pPopup->realX, pPopup->realY, pPopup->lastSize.x, pPopup->lastSize.y);
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
const auto PFOCUSEDTI = getFocusedTextInput();
|
|
|
|
|
|
|
|
if (!PFOCUSEDTI || !PFOCUSEDTI->pWlrInput->focused_surface)
|
|
|
|
return;
|
|
|
|
|
|
|
|
bool cursorRect = PFOCUSEDTI->pWlrInput->current.features & WLR_TEXT_INPUT_V3_FEATURE_CURSOR_RECTANGLE;
|
|
|
|
const auto PFOCUSEDSURFACE = PFOCUSEDTI->pWlrInput->focused_surface;
|
|
|
|
auto cursorBox = PFOCUSEDTI->pWlrInput->current.cursor_rectangle;
|
|
|
|
|
|
|
|
Vector2D parentPos;
|
|
|
|
Vector2D parentSize;
|
|
|
|
|
|
|
|
if (wlr_surface_is_layer_surface(PFOCUSEDSURFACE)) {
|
|
|
|
const auto PLS = g_pCompositor->getLayerSurfaceFromWlr(wlr_layer_surface_v1_from_wlr_surface(PFOCUSEDSURFACE));
|
|
|
|
|
|
|
|
if (PLS) {
|
|
|
|
parentPos = Vector2D(PLS->geometry.x, PLS->geometry.y) + g_pCompositor->getMonitorFromID(PLS->monitorID)->vecPosition;
|
|
|
|
parentSize = Vector2D(PLS->geometry.width, PLS->geometry.height);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const auto PWINDOW = g_pCompositor->getWindowFromSurface(PFOCUSEDSURFACE);
|
|
|
|
|
|
|
|
if (PWINDOW) {
|
|
|
|
parentPos = PWINDOW->m_vRealPosition.goalv();
|
|
|
|
parentSize = PWINDOW->m_vRealSize.goalv();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cursorRect) {
|
|
|
|
cursorBox = {0, 0, (int)parentSize.x, (int)parentSize.y};
|
|
|
|
}
|
|
|
|
|
|
|
|
// todo: anti-overflow
|
|
|
|
|
|
|
|
wlr_box finalBox = cursorBox;
|
|
|
|
|
|
|
|
pPopup->x = finalBox.x;
|
2022-08-05 18:54:59 +02:00
|
|
|
pPopup->y = finalBox.y + finalBox.height;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
|
|
|
pPopup->realX = finalBox.x + parentPos.x;
|
2022-08-05 18:54:59 +02:00
|
|
|
pPopup->realY = finalBox.y + parentPos.y + finalBox.height;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2022-08-23 11:16:27 +02:00
|
|
|
pPopup->lastSize = Vector2D(pPopup->pSurface->surface->current.width, pPopup->pSurface->surface->current.height);
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
wlr_input_popup_surface_v2_send_text_input_rectangle(pPopup->pSurface, &finalBox);
|
|
|
|
|
|
|
|
damagePopup(pPopup);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::setIMEPopupFocus(SIMEPopup* pPopup, wlr_surface* pSurface) {
|
|
|
|
updateInputPopup(pPopup);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Events::listener_mapInputPopup(void* owner, void* data) {
|
|
|
|
const auto PPOPUP = (SIMEPopup*)owner;
|
|
|
|
|
|
|
|
Debug::log(LOG, "Mapped an IME Popup");
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.updateInputPopup(PPOPUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Events::listener_unmapInputPopup(void* owner, void* data) {
|
|
|
|
const auto PPOPUP = (SIMEPopup*)owner;
|
|
|
|
|
|
|
|
Debug::log(LOG, "Unmapped an IME Popup");
|
|
|
|
|
2022-08-23 11:16:27 +02:00
|
|
|
g_pHyprRenderer->damageBox(PPOPUP->realX, PPOPUP->realY, PPOPUP->lastSize.x, PPOPUP->lastSize.y);
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
g_pInputManager->m_sIMERelay.updateInputPopup(PPOPUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Events::listener_destroyInputPopup(void* owner, void* data) {
|
|
|
|
const auto PPOPUP = (SIMEPopup*)owner;
|
|
|
|
|
|
|
|
Debug::log(LOG, "Removed an IME Popup");
|
|
|
|
|
|
|
|
PPOPUP->hyprListener_commitPopup.removeCallback();
|
|
|
|
PPOPUP->hyprListener_destroyPopup.removeCallback();
|
|
|
|
PPOPUP->hyprListener_focusedSurfaceUnmap.removeCallback();
|
|
|
|
PPOPUP->hyprListener_mapPopup.removeCallback();
|
|
|
|
PPOPUP->hyprListener_unmapPopup.removeCallback();
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.removePopup(PPOPUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Events::listener_commitInputPopup(void* owner, void* data) {
|
|
|
|
const auto PPOPUP = (SIMEPopup*)owner;
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.updateInputPopup(PPOPUP);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::removePopup(SIMEPopup* pPopup) {
|
|
|
|
m_lIMEPopups.remove(*pPopup);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::damagePopup(SIMEPopup* pPopup) {
|
|
|
|
if (!pPopup->pSurface->mapped)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto PFOCUSEDTI = getFocusedTextInput();
|
|
|
|
|
|
|
|
if (!PFOCUSEDTI || !PFOCUSEDTI->pWlrInput->focused_surface)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Vector2D parentPos;
|
|
|
|
|
|
|
|
const auto PFOCUSEDSURFACE = PFOCUSEDTI->pWlrInput->focused_surface;
|
|
|
|
|
|
|
|
if (wlr_surface_is_layer_surface(PFOCUSEDSURFACE)) {
|
|
|
|
const auto PLS = g_pCompositor->getLayerSurfaceFromWlr(wlr_layer_surface_v1_from_wlr_surface(PFOCUSEDSURFACE));
|
|
|
|
|
|
|
|
if (PLS) {
|
|
|
|
parentPos = Vector2D(PLS->geometry.x, PLS->geometry.y) + g_pCompositor->getMonitorFromID(PLS->monitorID)->vecPosition;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const auto PWINDOW = g_pCompositor->getWindowFromSurface(PFOCUSEDSURFACE);
|
|
|
|
|
|
|
|
if (PWINDOW) {
|
|
|
|
parentPos = PWINDOW->m_vRealPosition.goalv();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_pHyprRenderer->damageSurface(pPopup->pSurface->surface, parentPos.x + pPopup->x, parentPos.y + pPopup->y);
|
|
|
|
}
|
|
|
|
|
2022-08-05 16:21:08 +02:00
|
|
|
SIMEKbGrab* CInputMethodRelay::getIMEKeyboardGrab(SKeyboard* pKeyboard) {
|
|
|
|
|
|
|
|
if (!m_pWLRIME)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
if (!m_pKeyboardGrab.get())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const auto VIRTKB = wlr_input_device_get_virtual_keyboard(pKeyboard->keyboard);
|
|
|
|
|
|
|
|
if (VIRTKB && (wl_resource_get_client(VIRTKB->resource) == wl_resource_get_client(m_pKeyboardGrab->pWlrKbGrab->resource)))
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return m_pKeyboardGrab.get();
|
|
|
|
}
|
|
|
|
|
2022-08-05 13:03:37 +02:00
|
|
|
STextInput* CInputMethodRelay::getFocusedTextInput() {
|
|
|
|
for (auto& ti : m_lTextInputs) {
|
|
|
|
if (ti.pWlrInput->focused_surface) {
|
|
|
|
return &ti;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
STextInput* CInputMethodRelay::getFocusableTextInput() {
|
|
|
|
for (auto& ti : m_lTextInputs) {
|
|
|
|
if (ti.pPendingSurface) {
|
|
|
|
return &ti;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::onNewTextInput(wlr_text_input_v3* pInput) {
|
|
|
|
createNewTextInput(pInput);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::createNewTextInput(wlr_text_input_v3* pInput) {
|
|
|
|
const auto PTEXTINPUT = &m_lTextInputs.emplace_back();
|
|
|
|
|
|
|
|
PTEXTINPUT->pWlrInput = pInput;
|
|
|
|
|
|
|
|
PTEXTINPUT->hyprListener_textInputEnable.initCallback(&pInput->events.enable, [](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PINPUT = (STextInput*)owner;
|
|
|
|
|
|
|
|
if (!g_pInputManager->m_sIMERelay.m_pWLRIME) {
|
|
|
|
Debug::log(ERR, "Enabling TextInput on no IME!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Debug::log(LOG, "Enable TextInput");
|
|
|
|
|
|
|
|
wlr_input_method_v2_send_activate(g_pInputManager->m_sIMERelay.m_pWLRIME);
|
|
|
|
g_pInputManager->m_sIMERelay.commitIMEState(PINPUT->pWlrInput);
|
|
|
|
|
|
|
|
}, PTEXTINPUT, "textInput");
|
|
|
|
|
|
|
|
PTEXTINPUT->hyprListener_textInputCommit.initCallback(&pInput->events.commit, [](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PINPUT = (STextInput*)owner;
|
|
|
|
|
|
|
|
if (!g_pInputManager->m_sIMERelay.m_pWLRIME) {
|
|
|
|
Debug::log(ERR, "Committing TextInput on no IME!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!PINPUT->pWlrInput->current_enabled) {
|
|
|
|
Debug::log(ERR, "Disabled TextInput commit?");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.commitIMEState(PINPUT->pWlrInput);
|
|
|
|
|
|
|
|
}, PTEXTINPUT, "textInput");
|
|
|
|
|
|
|
|
PTEXTINPUT->hyprListener_textInputDisable.initCallback(&pInput->events.disable, [](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PINPUT = (STextInput*)owner;
|
|
|
|
|
|
|
|
if (!g_pInputManager->m_sIMERelay.m_pWLRIME) {
|
|
|
|
Debug::log(ERR, "Disabling TextInput on no IME!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Debug::log(LOG, "Disable TextInput");
|
|
|
|
|
|
|
|
wlr_input_method_v2_send_deactivate(g_pInputManager->m_sIMERelay.m_pWLRIME);
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.commitIMEState(PINPUT->pWlrInput);
|
|
|
|
|
|
|
|
}, PTEXTINPUT, "textInput");
|
|
|
|
|
|
|
|
PTEXTINPUT->hyprListener_textInputDestroy.initCallback(&pInput->events.destroy, [](void* owner, void* data) {
|
|
|
|
|
|
|
|
const auto PINPUT = (STextInput*)owner;
|
|
|
|
|
|
|
|
if (!g_pInputManager->m_sIMERelay.m_pWLRIME) {
|
|
|
|
Debug::log(ERR, "Disabling TextInput on no IME!");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PINPUT->pWlrInput->current_enabled) {
|
|
|
|
wlr_input_method_v2_send_deactivate(g_pInputManager->m_sIMERelay.m_pWLRIME);
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.commitIMEState(PINPUT->pWlrInput);
|
|
|
|
}
|
|
|
|
|
2022-08-05 13:19:16 +02:00
|
|
|
g_pInputManager->m_sIMERelay.setPendingSurface(PINPUT, nullptr);
|
|
|
|
|
2022-08-05 13:03:37 +02:00
|
|
|
PINPUT->hyprListener_textInputCommit.removeCallback();
|
|
|
|
PINPUT->hyprListener_textInputDestroy.removeCallback();
|
|
|
|
PINPUT->hyprListener_textInputDisable.removeCallback();
|
|
|
|
PINPUT->hyprListener_textInputEnable.removeCallback();
|
|
|
|
|
|
|
|
g_pInputManager->m_sIMERelay.removeTextInput(PINPUT->pWlrInput);
|
|
|
|
|
|
|
|
}, PTEXTINPUT, "textInput");
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::removeTextInput(wlr_text_input_v3* pInput) {
|
|
|
|
m_lTextInputs.remove_if([&](const auto& other) { return other.pWlrInput == pInput; });
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::commitIMEState(wlr_text_input_v3* pInput) {
|
2022-08-05 17:19:49 +02:00
|
|
|
if (!m_pWLRIME)
|
|
|
|
return;
|
|
|
|
|
2022-08-05 16:21:08 +02:00
|
|
|
if (pInput->active_features & WLR_TEXT_INPUT_V3_FEATURE_SURROUNDING_TEXT)
|
|
|
|
wlr_input_method_v2_send_surrounding_text(m_pWLRIME, pInput->current.surrounding.text, pInput->current.surrounding.cursor, pInput->current.surrounding.anchor);
|
|
|
|
|
2022-08-05 13:03:37 +02:00
|
|
|
wlr_input_method_v2_send_text_change_cause(m_pWLRIME, pInput->current.text_change_cause);
|
2022-08-05 16:21:08 +02:00
|
|
|
|
|
|
|
if (pInput->active_features & WLR_TEXT_INPUT_V3_FEATURE_CONTENT_TYPE)
|
|
|
|
wlr_input_method_v2_send_content_type(m_pWLRIME, pInput->current.content_type.hint, pInput->current.content_type.purpose);
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
for (auto& p : m_lIMEPopups) {
|
|
|
|
updateInputPopup(&p);
|
|
|
|
}
|
|
|
|
|
2022-08-05 13:03:37 +02:00
|
|
|
wlr_input_method_v2_send_done(m_pWLRIME);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::onKeyboardFocus(wlr_surface* pSurface) {
|
|
|
|
if (!m_pWLRIME)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (auto& ti : m_lTextInputs) {
|
|
|
|
if (ti.pPendingSurface) {
|
|
|
|
|
|
|
|
if (pSurface != ti.pPendingSurface)
|
2022-08-05 13:19:16 +02:00
|
|
|
setPendingSurface(&ti, nullptr);
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
} else if (ti.pWlrInput->focused_surface) {
|
|
|
|
|
|
|
|
if (pSurface != ti.pWlrInput->focused_surface) {
|
|
|
|
wlr_input_method_v2_send_deactivate(m_pWLRIME);
|
|
|
|
commitIMEState(ti.pWlrInput);
|
|
|
|
|
|
|
|
wlr_text_input_v3_send_leave(ti.pWlrInput);
|
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pSurface && wl_resource_get_client(ti.pWlrInput->resource) == wl_resource_get_client(pSurface->resource)) {
|
|
|
|
|
|
|
|
if (m_pWLRIME) {
|
|
|
|
wlr_text_input_v3_send_enter(ti.pWlrInput, pSurface);
|
|
|
|
} else {
|
2022-08-05 13:19:16 +02:00
|
|
|
setPendingSurface(&ti, pSurface);
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2022-08-05 13:19:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::setPendingSurface(STextInput* pInput, wlr_surface* pSurface) {
|
|
|
|
pInput->pPendingSurface = pSurface;
|
|
|
|
|
|
|
|
if (pSurface) {
|
|
|
|
pInput->hyprListener_pendingSurfaceDestroy.initCallback(&pSurface->events.destroy, [](void* owner, void* data) {
|
|
|
|
const auto PINPUT = (STextInput*)owner;
|
|
|
|
|
|
|
|
PINPUT->pPendingSurface = nullptr;
|
|
|
|
|
|
|
|
PINPUT->hyprListener_pendingSurfaceDestroy.removeCallback();
|
|
|
|
}, pInput, "TextInput");
|
|
|
|
} else {
|
|
|
|
pInput->hyprListener_pendingSurfaceDestroy.removeCallback();
|
|
|
|
}
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|