Files
ueberzugpp/plugins/wayfire/ueberzugpp.cpp
2025-12-28 00:36:34 -08:00

151 lines
4.9 KiB
C++

#include <memory>
#include <unordered_map>
#include <wayfire/core.hpp>
#include <wayfire/scene-render.hpp>
#include <wayfire/util.hpp>
#include <wayfire/view.hpp>
#include <wayfire/output.hpp>
#include <wayfire/plugin.hpp>
#include <wayfire/scene-operations.hpp>
#include <wayfire/plugins/ipc/ipc-method-repository.hpp>
#include <wayfire/plugins/common/shared-core-data.hpp>
#include <wayfire/unstable/wlr-view-events.hpp>
#include <wayfire/unstable/wlr-surface-node.hpp>
#include <wayfire/unstable/translation-node.hpp>
namespace wf
{
class ueberzugpp_surface_t
{
std::weak_ptr<wf::view_interface_t> terminal;
std::shared_ptr<scene::wlr_surface_node_t> surface;
std::shared_ptr<scene::translation_node_t> translation;
wlr_xdg_toplevel *toplevel;
wf::wl_listener_wrapper on_toplevel_destroy;
public:
ueberzugpp_surface_t(const std::weak_ptr<wf::view_interface_t>& _terminal,
wlr_xdg_toplevel *_toplevel):
terminal(_terminal),
toplevel(_toplevel)
{
// We create a custom surface node for the ueberzugpp window and wrap it in a translation node, so that
// we can control its position.
// We add the translation node as a subsurface of the terminal.
surface = std::make_shared<scene::wlr_surface_node_t>(_toplevel->base->surface, true);
translation = std::make_shared<scene::translation_node_t>();
translation->set_children_list({surface});
auto term = _terminal.lock();
wf::scene::add_front(term->get_surface_root_node(), translation);
// Finally, wait for the toplevel to be destroyed
on_toplevel_destroy.set_callback([this] (auto)
{
destroy_callback();
});
on_toplevel_destroy.connect(&toplevel->base->events.destroy);
}
std::function<void()> destroy_callback;
~ueberzugpp_surface_t()
{
wf::scene::remove_child(translation);
}
void set_offset(int xcoord, int ycoord)
{
translation->set_offset({xcoord, ycoord});
const auto term = terminal.lock();
if (!term) {
return;
}
term->damage();
}
};
class ueberzugpp_mapper : public wf::plugin_interface_t
{
public:
void init() override
{
ipc_repo->register_method("ueberzugpp/set_offset", set_offset);
wf::get_core().connect(&on_new_xdg_surface);
}
void fini() override
{
ipc_repo->unregister_method("ueberzugpp/set_offset");
}
ipc::method_callback set_offset = [this] (nlohmann::json json)
{
// NOLINTBEGIN
WFJSON_EXPECT_FIELD(json, "app-id", string);
WFJSON_EXPECT_FIELD(json, "x", number_integer);
WFJSON_EXPECT_FIELD(json, "y", number_integer);
// NOLINTEND
const std::string app_id = json["app-id"];
const int xcoord = json["x"];
const int ycoord = json["y"];
const auto search = surfaces.find(app_id);
if (search == surfaces.end())
{
return ipc::json_error("Unknown ueberzugpp window with appid " + app_id);
}
search->second->set_offset(xcoord, ycoord);
return ipc::json_ok();
};
shared_data::ref_ptr_t<ipc::method_repository_t> ipc_repo;
// When a new xdg_toplevel is created, we need to check whether it is an ueberzugpp window by looking at
// its app-id. If it is indeed an ueberzugpp window, we take over the toplevel (by setting
// use_default_implementation=false) and create our own ueberzugpp_surface. In addition, we directly map
// the surface to the currently focused view, if it exists.
wf::signal::connection_t<new_xdg_surface_signal> on_new_xdg_surface = [this] (new_xdg_surface_signal *event)
{
if (!event->use_default_implementation) {
return;
}
if (event->surface->role != WLR_XDG_SURFACE_ROLE_TOPLEVEL) {
return;
}
const auto* toplevel_title = event->surface->toplevel->title;
const std::string appid = toplevel_title != nullptr ? toplevel_title : "null";
const std::string search_for = "ueberzugpp_";
if (appid.find(search_for) != std::string::npos)
{
auto terminal = wf::get_core().get_active_output()->get_active_view();
if (!terminal)
{
LOGE("Which window to map ueberzugpp to????");
return;
}
event->use_default_implementation = false;
const auto [iter, was_inserted] = surfaces.insert_or_assign(appid,
std::make_unique<ueberzugpp_surface_t>(terminal->weak_from_this(), event->surface->toplevel));
iter->second->destroy_callback = [this, appid]
{
surfaces.erase(appid);
};
}
};
std::unordered_map<std::string, std::unique_ptr<ueberzugpp_surface_t>> surfaces;
};
} // end namespace wf
// NOLINTNEXTLINE
DECLARE_WAYFIRE_PLUGIN(wf::ueberzugpp_mapper);