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
|
|
|
|
2023-02-19 21:54:53 +01:00
|
|
|
CInputMethodRelay::CInputMethodRelay() {
|
2023-10-21 15:52:43 +02:00
|
|
|
g_pHookSystem->hookDynamic("keyboardFocus", [&](void* self, SCallbackInfo& info, std::any param) { onKeyboardFocus(std::any_cast<wlr_surface*>(param)); });
|
2023-02-19 21:54:53 +01:00
|
|
|
}
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
hyprListener_IMECommit.initCallback(
|
|
|
|
&m_pWLRIME->events.commit,
|
|
|
|
[&](void* owner, void* data) {
|
|
|
|
const auto PTI = getFocusedTextInput();
|
|
|
|
const auto PIMR = (CInputMethodRelay*)owner;
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
if (!PTI) {
|
|
|
|
Debug::log(LOG, "No focused TextInput on IME Commit");
|
|
|
|
return;
|
|
|
|
}
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
PTI->updateIMEState(PIMR->m_pWLRIME);
|
2022-12-16 18:17:31 +01:00
|
|
|
},
|
|
|
|
this, "IMERelay");
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
hyprListener_IMEDestroy.initCallback(
|
|
|
|
&m_pWLRIME->events.destroy,
|
|
|
|
[&](void* owner, void* data) {
|
|
|
|
m_pWLRIME = nullptr;
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
hyprListener_IMEDestroy.removeCallback();
|
|
|
|
hyprListener_IMECommit.removeCallback();
|
|
|
|
hyprListener_IMEGrab.removeCallback();
|
|
|
|
hyprListener_IMENewPopup.removeCallback();
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
const auto PTI = getFocusedTextInput();
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
Debug::log(LOG, "IME Destroy");
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2024-03-19 16:54:33 +01:00
|
|
|
if (PTI)
|
2024-03-23 00:08:52 +01:00
|
|
|
PTI->enter(PTI->focusedSurface());
|
2022-12-16 18:17:31 +01:00
|
|
|
},
|
|
|
|
this, "IMERelay");
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
hyprListener_IMEGrab.initCallback(
|
|
|
|
&m_pWLRIME->events.grab_keyboard,
|
|
|
|
[&](void* owner, void* data) {
|
|
|
|
Debug::log(LOG, "IME TextInput Keyboard Grab new");
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab = std::make_unique<SIMEKbGrab>();
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab->pKeyboard = wlr_seat_get_keyboard(g_pCompositor->m_sSeat.seat);
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
const auto PKBGRAB = (wlr_input_method_keyboard_grab_v2*)data;
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab->pWlrKbGrab = PKBGRAB;
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
wlr_input_method_keyboard_grab_v2_set_keyboard(m_pKeyboardGrab->pWlrKbGrab, m_pKeyboardGrab->pKeyboard);
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab->hyprListener_grabDestroy.initCallback(
|
|
|
|
&PKBGRAB->events.destroy,
|
|
|
|
[&](void* owner, void* data) {
|
|
|
|
m_pKeyboardGrab->hyprListener_grabDestroy.removeCallback();
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
Debug::log(LOG, "IME TextInput Keyboard Grab destroy");
|
2022-08-05 16:21:08 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
m_pKeyboardGrab.reset(nullptr);
|
|
|
|
},
|
|
|
|
m_pKeyboardGrab.get(), "IME Keyboard Grab");
|
|
|
|
},
|
|
|
|
this, "IMERelay");
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
hyprListener_IMENewPopup.initCallback(
|
|
|
|
&m_pWLRIME->events.new_popup_surface,
|
|
|
|
[&](void* owner, void* data) {
|
|
|
|
const auto PNEWPOPUP = &m_lIMEPopups.emplace_back();
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
PNEWPOPUP->pSurface = (wlr_input_popup_surface_v2*)data;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
PNEWPOPUP->hyprListener_commitPopup.initCallback(&PNEWPOPUP->pSurface->surface->events.commit, &Events::listener_commitInputPopup, PNEWPOPUP, "IME Popup");
|
2023-06-03 12:20:23 +02:00
|
|
|
PNEWPOPUP->hyprListener_mapPopup.initCallback(&PNEWPOPUP->pSurface->surface->events.map, &Events::listener_mapInputPopup, PNEWPOPUP, "IME Popup");
|
|
|
|
PNEWPOPUP->hyprListener_unmapPopup.initCallback(&PNEWPOPUP->pSurface->surface->events.unmap, &Events::listener_unmapInputPopup, PNEWPOPUP, "IME Popup");
|
2022-12-16 18:17:31 +01:00
|
|
|
PNEWPOPUP->hyprListener_destroyPopup.initCallback(&PNEWPOPUP->pSurface->events.destroy, &Events::listener_destroyInputPopup, PNEWPOPUP, "IME Popup");
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
Debug::log(LOG, "New input popup");
|
|
|
|
},
|
|
|
|
this, "IMERelay");
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-19 16:54:33 +01:00
|
|
|
if (const auto PTI = getFocusedTextInput(); PTI)
|
2024-03-23 00:08:52 +01:00
|
|
|
PTI->enter(PTI->focusedSurface());
|
2023-03-14 13:57:50 +01:00
|
|
|
}
|
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
void CInputMethodRelay::updateInputPopup(SIMEPopup* pPopup) {
|
2023-06-03 12:20:23 +02:00
|
|
|
if (!pPopup->pSurface->surface->mapped)
|
2022-08-05 17:07:01 +02:00
|
|
|
return;
|
|
|
|
|
2022-08-23 11:16:27 +02:00
|
|
|
// damage last known pos & size
|
2024-03-23 00:08:52 +01:00
|
|
|
g_pHyprRenderer->damageBox(&pPopup->lastBox);
|
2022-08-23 11:16:27 +02:00
|
|
|
|
2022-08-05 17:07:01 +02:00
|
|
|
const auto PFOCUSEDTI = getFocusedTextInput();
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (!PFOCUSEDTI || !PFOCUSEDTI->focusedSurface())
|
2022-08-05 17:07:01 +02:00
|
|
|
return;
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
bool cursorRect = PFOCUSEDTI->hasCursorRectangle();
|
|
|
|
const auto PFOCUSEDSURFACE = PFOCUSEDTI->focusedSurface();
|
|
|
|
CBox cursorBox = PFOCUSEDTI->cursorBox();
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
CBox parentBox;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
const auto PSURFACE = CWLSurface::surfaceFromWlr(PFOCUSEDSURFACE);
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (!PSURFACE)
|
|
|
|
parentBox = {0, 0, 200, 200};
|
|
|
|
else
|
|
|
|
parentBox = PSURFACE->getSurfaceBoxGlobal().value_or(CBox{0, 0, 200, 200});
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (!cursorRect)
|
|
|
|
cursorBox = {0, 0, (int)parentBox.w, (int)parentBox.h};
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
CMonitor* pMonitor = g_pCompositor->getMonitorFromVector(cursorBox.middle());
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (cursorBox.y + parentBox.y + pPopup->pSurface->surface->current.height + cursorBox.height > pMonitor->vecPosition.y + pMonitor->vecSize.y)
|
|
|
|
cursorBox.y -= pPopup->pSurface->surface->current.height + cursorBox.height;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (cursorBox.x + parentBox.x + pPopup->pSurface->surface->current.width > pMonitor->vecPosition.x + pMonitor->vecSize.x)
|
|
|
|
cursorBox.x -= (cursorBox.x + parentBox.x + pPopup->pSurface->surface->current.width) - (pMonitor->vecPosition.x + pMonitor->vecSize.x);
|
2023-03-14 17:12:25 +01:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
pPopup->x = cursorBox.x;
|
|
|
|
pPopup->y = cursorBox.y + cursorBox.height;
|
2023-03-14 17:12:25 +01:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
pPopup->realX = cursorBox.x + parentBox.x;
|
|
|
|
pPopup->realY = cursorBox.y + parentBox.y + cursorBox.height;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
pPopup->lastBox = cursorBox;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
wlr_input_popup_surface_v2_send_text_input_rectangle(pPopup->pSurface, cursorBox.pWlr());
|
2022-08-05 17:07:01 +02:00
|
|
|
|
|
|
|
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);
|
2023-01-10 19:21:59 +01:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (const auto PMONITOR = g_pCompositor->getMonitorFromVector(PPOPUP->lastBox.middle()); PMONITOR)
|
2023-01-10 19:21:59 +01:00
|
|
|
wlr_surface_send_enter(PPOPUP->pSurface->surface, PMONITOR->output);
|
2022-08-05 17:07:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Events::listener_unmapInputPopup(void* owner, void* data) {
|
|
|
|
const auto PPOPUP = (SIMEPopup*)owner;
|
|
|
|
|
|
|
|
Debug::log(LOG, "Unmapped an IME Popup");
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
g_pHyprRenderer->damageBox(&PPOPUP->lastBox);
|
2022-08-23 11:16:27 +02:00
|
|
|
|
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) {
|
2023-06-03 12:20:23 +02:00
|
|
|
if (!pPopup->pSurface->surface->mapped)
|
2022-08-05 17:07:01 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
const auto PFOCUSEDTI = getFocusedTextInput();
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (!PFOCUSEDTI || !PFOCUSEDTI->focusedSurface())
|
2022-08-05 17:07:01 +02:00
|
|
|
return;
|
|
|
|
|
2022-12-16 18:17:31 +01:00
|
|
|
Vector2D parentPos;
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
const auto PFOCUSEDSURFACE = PFOCUSEDTI->focusedSurface();
|
2022-08-05 17:07:01 +02:00
|
|
|
|
2023-02-03 13:43:43 +01:00
|
|
|
if (wlr_layer_surface_v1_try_from_wlr_surface(PFOCUSEDSURFACE)) {
|
|
|
|
const auto PLS = g_pCompositor->getLayerSurfaceFromWlr(wlr_layer_surface_v1_try_from_wlr_surface(PFOCUSEDSURFACE));
|
2022-08-05 17:07:01 +02:00
|
|
|
|
|
|
|
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) {
|
2024-03-02 01:35:17 +01:00
|
|
|
parentPos = PWINDOW->m_vRealPosition.goal();
|
2022-08-05 17:07:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
CTextInput* CInputMethodRelay::getFocusedTextInput() {
|
|
|
|
if (!g_pCompositor->m_pLastFocus)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
for (auto& ti : m_vTextInputs) {
|
|
|
|
if (ti->focusedSurface() == g_pCompositor->m_pLastFocus)
|
|
|
|
return ti.get();
|
|
|
|
}
|
2022-08-05 13:03:37 +02:00
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::onNewTextInput(wlr_text_input_v3* pInput) {
|
2024-03-23 00:08:52 +01:00
|
|
|
m_vTextInputs.emplace_back(std::make_unique<CTextInput>(pInput));
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
void CInputMethodRelay::onNewTextInput(STextInputV1* pTIV1) {
|
|
|
|
m_vTextInputs.emplace_back(std::make_unique<CTextInput>(pTIV1));
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
void CInputMethodRelay::removeTextInput(CTextInput* pInput) {
|
|
|
|
m_vTextInputs.remove_if([&](const auto& other) { return other.get() == pInput; });
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
void CInputMethodRelay::commitIMEState(CTextInput* pInput) {
|
2022-08-05 17:19:49 +02:00
|
|
|
if (!m_pWLRIME)
|
|
|
|
return;
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
pInput->commitStateToIME(m_pWLRIME);
|
2022-08-05 13:03:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CInputMethodRelay::onKeyboardFocus(wlr_surface* pSurface) {
|
|
|
|
if (!m_pWLRIME)
|
|
|
|
return;
|
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (pSurface == m_pLastKbFocus)
|
2024-03-19 16:54:33 +01:00
|
|
|
return;
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
m_pLastKbFocus = pSurface;
|
2023-03-14 13:57:50 +01:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
for (auto& ti : m_vTextInputs) {
|
|
|
|
if (!ti->focusedSurface())
|
|
|
|
continue;
|
2022-08-05 13:03:37 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
ti->leave();
|
2024-03-19 16:54:33 +01:00
|
|
|
}
|
2022-08-05 13:19:16 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
for (auto& ti : m_vTextInputs) {
|
|
|
|
if (!ti->isV3())
|
|
|
|
continue;
|
2024-03-22 19:45:24 +01:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
if (ti->client() != wl_resource_get_client(pSurface->resource))
|
|
|
|
continue;
|
2022-08-05 13:19:16 +02:00
|
|
|
|
2024-03-23 00:08:52 +01:00
|
|
|
ti->enter(pSurface);
|
2022-08-05 13:19:16 +02:00
|
|
|
}
|
2022-09-25 20:07:48 +02:00
|
|
|
}
|