Hyprland/src/managers/input/TextInput.cpp

308 lines
10 KiB
C++
Raw Normal View History

#include "TextInput.hpp"
#include "InputManager.hpp"
#include "../../protocols/TextInputV1.hpp"
#include "../../desktop/state/FocusState.hpp"
2024-04-25 23:27:25 +01:00
#include "../../protocols/TextInputV3.hpp"
2024-05-01 16:41:17 +01:00
#include "../../protocols/InputMethodV2.hpp"
#include "../../protocols/core/Compositor.hpp"
CTextInput::CTextInput(WP<CTextInputV1> ti) : m_v1Input(ti) {
initCallbacks();
}
CTextInput::CTextInput(WP<CTextInputV3> ti) : m_v3Input(ti) {
initCallbacks();
}
void CTextInput::initCallbacks() {
2024-04-25 23:27:25 +01:00
if (isV3()) {
const auto INPUT = m_v3Input.lock();
m_listeners.enable = INPUT->m_events.enable.listen([this] { onEnabled(); });
m_listeners.disable = INPUT->m_events.disable.listen([this] { onDisabled(); });
m_listeners.commit = INPUT->m_events.onCommit.listen([this] { onCommit(); });
m_listeners.reset = INPUT->m_events.reset.listen([this] { onReset(); });
m_listeners.destroy = INPUT->m_events.destroy.listen([this] {
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
g_pInputManager->m_relay.removeTextInput(this);
if (!g_pInputManager->m_relay.getFocusedTextInput())
g_pInputManager->m_relay.deactivateIME(this);
2024-04-25 23:27:25 +01:00
});
if (Desktop::focusState()->surface() && Desktop::focusState()->surface()->client() == INPUT->client())
enter(Desktop::focusState()->surface());
2024-04-25 23:27:25 +01:00
} else {
const auto INPUT = m_v1Input.lock();
2024-04-25 23:27:25 +01:00
m_listeners.enable = INPUT->m_events.enable.listen([this](const auto& surface) { onEnabled(surface); });
m_listeners.disable = INPUT->m_events.disable.listen([this] { onDisabled(); });
m_listeners.commit = INPUT->m_events.onCommit.listen([this] { onCommit(); });
m_listeners.reset = INPUT->m_events.reset.listen([this] { onReset(); });
m_listeners.destroy = INPUT->m_events.destroy.listen([this] {
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
g_pInputManager->m_relay.removeTextInput(this);
if (!g_pInputManager->m_relay.getFocusedTextInput())
g_pInputManager->m_relay.deactivateIME(this);
});
2024-04-25 23:27:25 +01:00
}
}
void CTextInput::onEnabled(SP<CWLSurfaceResource> surfV1) {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::DEBUG, "TI ENABLE");
if (g_pInputManager->m_relay.m_inputMethod.expired()) {
2025-12-17 20:08:47 +00:00
// Log::logger->log(Log::WARN, "Enabling TextInput on no IME!");
return;
}
// v1 only, map surface to PTI
if (!isV3()) {
if (Desktop::focusState()->surface() != surfV1 || !m_v1Input->m_active)
return;
enter(surfV1);
}
g_pInputManager->m_relay.activateIME(this);
}
void CTextInput::onDisabled() {
if (g_pInputManager->m_relay.m_inputMethod.expired()) {
2025-12-17 20:08:47 +00:00
// Log::logger->log(Log::WARN, "Disabling TextInput on no IME!");
return;
}
if (!isV3())
2024-03-28 02:28:16 +00:00
leave();
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
if (!focusedSurface())
return;
const auto PFOCUSEDTI = g_pInputManager->m_relay.getFocusedTextInput();
if (!PFOCUSEDTI || PFOCUSEDTI != this)
return;
g_pInputManager->m_relay.deactivateIME(this);
}
void CTextInput::onReset() {
if (g_pInputManager->m_relay.m_inputMethod.expired())
return;
if (!focusedSurface())
return;
const auto PFOCUSEDTI = g_pInputManager->m_relay.getFocusedTextInput();
if (!PFOCUSEDTI || PFOCUSEDTI != this)
return;
g_pInputManager->m_relay.deactivateIME(this, false);
g_pInputManager->m_relay.activateIME(this);
}
void CTextInput::onCommit() {
if (g_pInputManager->m_relay.m_inputMethod.expired()) {
2025-12-17 20:08:47 +00:00
// Log::logger->log(Log::WARN, "Committing TextInput on no IME!");
return;
}
if (!(isV3() ? m_v3Input->m_current.enabled.value : m_v1Input->m_active)) {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::WARN, "Disabled TextInput commit?");
return;
}
g_pInputManager->m_relay.commitIMEState(this);
}
void CTextInput::setFocusedSurface(SP<CWLSurfaceResource> pSurface) {
if (pSurface == m_focusedSurface)
return;
m_focusedSurface = pSurface;
if (!pSurface)
return;
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
m_listeners.surfaceUnmap = pSurface->m_events.unmap.listen([this] {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::DEBUG, "Unmap TI owner1");
if (m_enterLocks)
m_enterLocks--;
m_focusedSurface.reset();
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
if (isV3() && !m_v3Input.expired() && m_v3Input->m_current.enabled.value) {
m_v3Input->m_pending.enabled.value = false;
m_v3Input->m_pending.enabled.isDisablePending = false;
m_v3Input->m_pending.enabled.isEnablePending = false;
m_v3Input->m_current.enabled.value = false;
2024-09-13 01:41:24 +09:00
}
if (!g_pInputManager->m_relay.getFocusedTextInput())
g_pInputManager->m_relay.deactivateIME(this);
});
m_listeners.surfaceDestroy = pSurface->m_events.destroy.listen([this] {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::DEBUG, "Destroy TI owner1");
if (m_enterLocks)
m_enterLocks--;
m_focusedSurface.reset();
m_listeners.surfaceUnmap.reset();
m_listeners.surfaceDestroy.reset();
if (isV3() && !m_v3Input.expired() && m_v3Input->m_current.enabled.value) {
m_v3Input->m_pending.enabled.value = false;
m_v3Input->m_pending.enabled.isDisablePending = false;
m_v3Input->m_pending.enabled.isEnablePending = false;
m_v3Input->m_current.enabled.value = false;
2024-09-13 01:41:24 +09:00
}
if (!g_pInputManager->m_relay.getFocusedTextInput())
g_pInputManager->m_relay.deactivateIME(this);
});
}
bool CTextInput::isV3() {
return m_v3Input && !m_v1Input;
}
void CTextInput::enter(SP<CWLSurfaceResource> pSurface) {
if (!pSurface || !pSurface->m_mapped)
return;
if (pSurface == focusedSurface())
return;
if (focusedSurface())
leave();
m_enterLocks++;
if (m_enterLocks != 1) {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::ERR, "BUG THIS: TextInput has != 1 locks in enter");
leave();
m_enterLocks = 1;
}
2024-04-25 23:27:25 +01:00
if (isV3())
m_v3Input->enter(pSurface);
else {
m_v1Input->enter(pSurface);
}
setFocusedSurface(pSurface);
}
void CTextInput::leave() {
if (!focusedSurface())
return;
m_enterLocks--;
if (m_enterLocks != 0) {
2025-12-17 20:08:47 +00:00
Log::logger->log(Log::ERR, "BUG THIS: TextInput has != 0 locks in leave");
m_enterLocks = 0;
}
2024-09-13 01:41:24 +09:00
if (isV3())
m_v3Input->leave(focusedSurface());
2024-09-13 01:41:24 +09:00
else
m_v1Input->leave();
setFocusedSurface(nullptr);
g_pInputManager->m_relay.deactivateIME(this);
}
SP<CWLSurfaceResource> CTextInput::focusedSurface() {
return m_focusedSurface.lock();
}
wl_client* CTextInput::client() {
return isV3() ? m_v3Input->client() : m_v1Input->client();
}
2024-05-01 16:41:17 +01:00
void CTextInput::commitStateToIME(SP<CInputMethodV2> ime) {
if (isV3() && !m_v3Input.expired()) {
const auto INPUT = m_v3Input.lock();
2024-04-25 23:27:25 +01:00
if (INPUT->m_current.surrounding.updated)
ime->surroundingText(INPUT->m_current.surrounding.text, INPUT->m_current.surrounding.cursor, INPUT->m_current.surrounding.anchor);
ime->textChangeCause(INPUT->m_current.cause);
if (INPUT->m_current.contentType.updated)
ime->textContentType(INPUT->m_current.contentType.hint, INPUT->m_current.contentType.purpose);
} else if (!m_v1Input.expired()) {
const auto INPUT = m_v1Input.lock();
if (INPUT->m_pendingSurrounding.isPending)
ime->surroundingText(INPUT->m_pendingSurrounding.text, INPUT->m_pendingSurrounding.cursor, INPUT->m_pendingSurrounding.anchor);
2024-05-01 16:41:17 +01:00
ime->textChangeCause(ZWP_TEXT_INPUT_V3_CHANGE_CAUSE_INPUT_METHOD);
if (m_v1Input->m_pendingContentType.isPending)
ime->textContentType(sc<zwpTextInputV3ContentHint>(INPUT->m_pendingContentType.hint), sc<zwpTextInputV3ContentPurpose>(INPUT->m_pendingContentType.purpose));
}
g_pInputManager->m_relay.updateAllPopups();
2024-05-01 16:41:17 +01:00
ime->done();
}
2024-05-01 16:41:17 +01:00
void CTextInput::updateIMEState(SP<CInputMethodV2> ime) {
if (isV3()) {
const auto INPUT = m_v3Input.lock();
if (ime->m_current.preeditString.committed)
INPUT->preeditString(ime->m_current.preeditString.string, ime->m_current.preeditString.begin, ime->m_current.preeditString.end);
if (ime->m_current.committedString.committed)
INPUT->commitString(ime->m_current.committedString.string);
2024-04-25 23:27:25 +01:00
if (ime->m_current.deleteSurrounding.committed)
INPUT->deleteSurroundingText(ime->m_current.deleteSurrounding.before, ime->m_current.deleteSurrounding.after);
2024-04-25 23:27:25 +01:00
INPUT->sendDone();
} else {
const auto INPUT = m_v1Input.lock();
if (ime->m_current.preeditString.committed) {
INPUT->preeditCursor(ime->m_current.preeditString.begin);
INPUT->preeditStyling(0, std::string(ime->m_current.preeditString.string).length(), ZWP_TEXT_INPUT_V1_PREEDIT_STYLE_HIGHLIGHT);
INPUT->preeditString(m_v1Input->m_serial, ime->m_current.preeditString.string.c_str(), "");
} else {
INPUT->preeditCursor(0);
INPUT->preeditStyling(0, 0, ZWP_TEXT_INPUT_V1_PREEDIT_STYLE_HIGHLIGHT);
INPUT->preeditString(m_v1Input->m_serial, "", "");
}
if (ime->m_current.committedString.committed)
INPUT->commitString(m_v1Input->m_serial, ime->m_current.committedString.string.c_str());
if (ime->m_current.deleteSurrounding.committed) {
INPUT->deleteSurroundingText(std::string(ime->m_current.preeditString.string).length() - ime->m_current.deleteSurrounding.before,
ime->m_current.deleteSurrounding.after + ime->m_current.deleteSurrounding.before);
if (ime->m_current.preeditString.committed)
INPUT->commitString(m_v1Input->m_serial, ime->m_current.preeditString.string.c_str());
}
}
}
bool CTextInput::hasCursorRectangle() {
return !isV3() || m_v3Input->m_current.box.updated;
}
CBox CTextInput::cursorBox() {
return CBox{isV3() ? m_v3Input->m_current.box.cursorBox : m_v1Input->m_cursorRectangle};
}