a18ccb06e6
The cordinate from text-input protocol is based on original surface, instead of the geometry based on any external protocol, e.g. xdg_surface.
175 lines
6 KiB
C++
175 lines
6 KiB
C++
/*
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
SPDX-FileCopyrightText: 2020 Aleix Pol Gonzalez <aleixpol@kde.org>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
#include "inputpanelv1client.h"
|
|
#include "deleted.h"
|
|
#include "wayland_server.h"
|
|
#include "workspace.h"
|
|
#include "abstract_wayland_output.h"
|
|
#include "platform.h"
|
|
#include "inputmethod.h"
|
|
#include <KWaylandServer/output_interface.h>
|
|
#include <KWaylandServer/seat_interface.h>
|
|
#include <KWaylandServer/surface_interface.h>
|
|
#include <KWaylandServer/textinput_v2_interface.h>
|
|
#include <KWaylandServer/textinput_v3_interface.h>
|
|
|
|
using namespace KWaylandServer;
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
InputPanelV1Client::InputPanelV1Client(InputPanelSurfaceV1Interface *panelSurface)
|
|
: WaylandClient(panelSurface->surface())
|
|
, m_panelSurface(panelSurface)
|
|
{
|
|
setSkipSwitcher(true);
|
|
setSkipPager(true);
|
|
setSkipTaskbar(true);
|
|
|
|
connect(surface(), &SurfaceInterface::aboutToBeDestroyed, this, &InputPanelV1Client::destroyClient);
|
|
connect(surface(), &SurfaceInterface::sizeChanged, this, &InputPanelV1Client::reposition);
|
|
connect(surface(), &SurfaceInterface::mapped, this, &InputPanelV1Client::updateDepth);
|
|
|
|
connect(panelSurface, &InputPanelSurfaceV1Interface::topLevel, this, &InputPanelV1Client::showTopLevel);
|
|
connect(panelSurface, &InputPanelSurfaceV1Interface::overlayPanel, this, &InputPanelV1Client::showOverlayPanel);
|
|
connect(panelSurface, &InputPanelSurfaceV1Interface::destroyed, this, &InputPanelV1Client::destroyClient);
|
|
|
|
InputMethod::self()->setPanel(this);
|
|
}
|
|
|
|
void InputPanelV1Client::showOverlayPanel()
|
|
{
|
|
setOutput(nullptr);
|
|
m_mode = Overlay;
|
|
reposition();
|
|
setReadyForPainting();
|
|
}
|
|
|
|
void InputPanelV1Client::showTopLevel(OutputInterface *output, InputPanelSurfaceV1Interface::Position position)
|
|
{
|
|
Q_UNUSED(position);
|
|
m_mode = Toplevel;
|
|
setOutput(output);
|
|
}
|
|
|
|
void InputPanelV1Client::allow()
|
|
{
|
|
setReadyForPainting();
|
|
reposition();
|
|
}
|
|
|
|
void KWin::InputPanelV1Client::reposition()
|
|
{
|
|
if (!readyForPainting()) {
|
|
return;
|
|
}
|
|
|
|
switch (m_mode) {
|
|
case Toplevel: {
|
|
if (m_output) {
|
|
const QSize panelSize = surface()->size();
|
|
if (!panelSize.isValid() || panelSize.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
QRect availableArea;
|
|
if (waylandServer()->isScreenLocked()) {
|
|
availableArea = m_output->geometry();
|
|
} else {
|
|
availableArea = workspace()->clientArea(MaximizeArea, this, m_output);
|
|
}
|
|
QRect geo(availableArea.topLeft(), panelSize);
|
|
geo.translate((availableArea.width() - panelSize.width())/2, availableArea.height() - panelSize.height());
|
|
moveResize(geo);
|
|
}
|
|
} break;
|
|
case Overlay: {
|
|
auto textInputSurface = waylandServer()->seat()->focusedTextInputSurface();
|
|
auto textClient = waylandServer()->findClient(textInputSurface);
|
|
QRect cursorRectangle;
|
|
auto textInputV2 = waylandServer()->seat()->textInputV2();
|
|
if (textInputV2 && textInputV2->isEnabled() && textInputV2->surface() == textInputSurface) {
|
|
cursorRectangle = textInputV2->cursorRectangle();
|
|
}
|
|
auto textInputV3 = waylandServer()->seat()->textInputV3();
|
|
if (textInputV3 && textInputV3->isEnabled() && textInputV3->surface() == textInputSurface) {
|
|
cursorRectangle = textInputV3->cursorRectangle();
|
|
}
|
|
if (textClient) {
|
|
cursorRectangle.translate(textClient->bufferGeometry().topLeft());
|
|
const QRect screen = Workspace::self()->clientArea(PlacementArea, cursorRectangle.bottomLeft(), 0);
|
|
|
|
// Reuse the similar logic like xdg popup
|
|
QRect popupRect(popupOffset(cursorRectangle, Qt::BottomEdge | Qt::LeftEdge, Qt::RightEdge | Qt::BottomEdge, surface()->size()), surface()->size());
|
|
|
|
if (popupRect.left() < screen.left()) {
|
|
popupRect.moveLeft(screen.left());
|
|
}
|
|
if (popupRect.right() > screen.right()) {
|
|
popupRect.moveRight(screen.right());
|
|
}
|
|
if (popupRect.top() < screen.top() || popupRect.bottom() > screen.bottom()) {
|
|
auto flippedPopupRect =
|
|
QRect(popupOffset(cursorRectangle, Qt::TopEdge | Qt::LeftEdge, Qt::RightEdge | Qt::TopEdge, surface()->size()), surface()->size());
|
|
|
|
// if it still doesn't fit we should continue with the unflipped version
|
|
if (flippedPopupRect.top() >= screen.top() || flippedPopupRect.bottom() <= screen.bottom()) {
|
|
popupRect.moveTop(flippedPopupRect.top());
|
|
}
|
|
}
|
|
moveResize(popupRect);
|
|
}
|
|
} break;
|
|
}
|
|
}
|
|
|
|
void InputPanelV1Client::destroyClient()
|
|
{
|
|
markAsZombie();
|
|
|
|
Deleted *deleted = Deleted::create(this);
|
|
Q_EMIT windowClosed(this, deleted);
|
|
StackingUpdatesBlocker blocker(workspace());
|
|
waylandServer()->removeClient(this);
|
|
deleted->unrefWindow();
|
|
|
|
delete this;
|
|
}
|
|
|
|
NET::WindowType InputPanelV1Client::windowType(bool, int) const
|
|
{
|
|
return NET::Utility;
|
|
}
|
|
|
|
QRect InputPanelV1Client::inputGeometry() const
|
|
{
|
|
return readyForPainting() ? surface()->input().boundingRect().translated(pos()) : QRect();
|
|
}
|
|
|
|
void InputPanelV1Client::setOutput(OutputInterface *outputIface)
|
|
{
|
|
if (m_output) {
|
|
disconnect(m_output, &AbstractWaylandOutput::geometryChanged, this, &InputPanelV1Client::reposition);
|
|
}
|
|
|
|
m_output = waylandServer()->findOutput(outputIface);
|
|
|
|
if (m_output) {
|
|
connect(m_output, &AbstractWaylandOutput::geometryChanged, this, &InputPanelV1Client::reposition);
|
|
}
|
|
}
|
|
|
|
void InputPanelV1Client::moveResizeInternal(const QRect &rect, MoveResizeMode mode)
|
|
{
|
|
Q_UNUSED(mode)
|
|
updateGeometry(rect);
|
|
}
|
|
|
|
} // namespace KWin
|