Added features to readme.

This commit is contained in:
Arron David Nelson 2024-07-03 19:49:18 -07:00
parent 7e380efd7d
commit fb86dca332
4 changed files with 226 additions and 22 deletions

View File

@ -215,7 +215,10 @@ elseif (IS_OS_LINUX)
if (LINUX_WINDOW_SYSTEM STREQUAL "Wayland") if (LINUX_WINDOW_SYSTEM STREQUAL "Wayland")
add_compile_definitions(EHS_WS_WAYLAND) add_compile_definitions(EHS_WS_WAYLAND)
list(APPEND EHS_SOURCES src/io/xdg-shell-protocol.c include/ehs/io/xdg-shell-client-protocol.h src/io/Window_Way.cpp include/ehs/io/Window_Way.h) list(APPEND EHS_SOURCES
src/io/xdg-shell-protocol.c include/ehs/io/xdg-shell-client-protocol.h
src/io/xdg-decoration.c include/ehs/io/xdg-decoration.h
src/io/Window_Way.cpp include/ehs/io/Window_Way.h)
message("Building for Wayland.") message("Building for Wayland.")
elseif (LINUX_WINDOW_SYSTEM STREQUAL "XCB") elseif (LINUX_WINDOW_SYSTEM STREQUAL "XCB")
add_compile_definitions(EHS_WS_XCB) add_compile_definitions(EHS_WS_XCB)

View File

@ -1888,11 +1888,7 @@ namespace ehs
if (aSize != bSize) if (aSize != bSize)
return false; return false;
for (UInt_64 i = 0; i < aSize; ++i) return Util::Compare(a, b, aSize);
if (a[i] != b[i])
return false;
return true;
} }
}; };

View File

@ -4,19 +4,28 @@
#include <wayland-client.h> #include <wayland-client.h>
#include "xdg-shell-client-protocol.h" #include "xdg-shell-client-protocol.h"
#include "xdg-decoration.h"
namespace ehs namespace ehs
{ {
class Window : public BaseWindow class Window : public BaseWindow
{ {
private: protected:
wl_display* display; wl_display *display;
wl_registry *registry; wl_registry *registry;
wl_compositor* compositor; wl_compositor *compositor;
wl_surface* surface; wl_surface *wlSurface;
xdg_wm_base *xdgShell; xdg_wm_base *xdgShell;
xdg_surface *xdgSurface; xdg_surface *xdgSurface;
xdg_toplevel *xdgToplevel; xdg_toplevel *xdgToplevel;
wl_shm *shm;
wl_buffer *buffer;
zxdg_decoration_manager_v1 *decManager;
zxdg_toplevel_decoration_v1 *dec;
static Int_32 CreateShmFile(Size size);
void CreateBuffer(Vec2_u32 scale);
static void SurfaceConfigure(void *data, xdg_surface *xdg_surface, UInt_32 serial); static void SurfaceConfigure(void *data, xdg_surface *xdg_surface, UInt_32 serial);
@ -26,11 +35,21 @@ namespace ehs
static void RegistryRemoved(void *data, wl_registry *registry, UInt_32 id); static void RegistryRemoved(void *data, wl_registry *registry, UInt_32 id);
static void Resolution(void *data, struct xdg_toplevel *xdg_toplevel, Int_32 width, Int_32 height, wl_array *states);
public: public:
~Window() override; ~Window() override;
Window(); Window();
Window(Window &&win) noexcept;
Window(const Window &win);
Window &operator=(Window &&win) noexcept;
Window &operator=(const Window &win);
void Create_32(const Str_32& title, const Vec2_s32& pos, const Vec2_u32 scale) override; void Create_32(const Str_32& title, const Vec2_s32& pos, const Vec2_u32 scale) override;
void Create_16(const Str_16& title, const Vec2_s32& pos, const Vec2_u32 scale) override; void Create_16(const Str_16& title, const Vec2_s32& pos, const Vec2_u32 scale) override;

View File

@ -1,7 +1,74 @@
#include "ehs/io/Window_Way.h" #include "ehs/io/Window_Way.h"
#include <unistd.h>
#include <sys/mman.h>
#include "ehs/io/Console.h"
#include "ehs/io/xdg-shell-client-protocol.h"
namespace ehs namespace ehs
{ {
Int_32 Window::CreateShmFile(Size size)
{
const char *path = getenv("XDG_RUNTIME_DIR");
if (!path)
{
EHS_LOG_INT(LogType::ERR, 0, "XDG_RUNTIME_DIR is not set.");
return -1;
}
Str_8 tmplPath = Str_8(path) + "/wl_shm_XXXXXX";
Int_32 fd = mkstemp(tmplPath);
if (fd < 0)
{
EHS_LOG_INT(LogType::ERR, 1, "Failed to create shared memory file.");
return -1;
}
if (unlink(tmplPath) < 0)
{
EHS_LOG_INT(LogType::ERR, 2, "Failed to unlink shared memory file.");
close(fd);
return -1;
}
if (ftruncate(fd, size) < 0)
{
EHS_LOG_INT(LogType::ERR, 3, "Failed to set size of shared memory file to " + Str_8::FromNum(size) + ".");
close(fd);
return -1;
}
EHS_LOG_SUCCESS();
return fd;
}
void Window::CreateBuffer(Vec2_u32 scale)
{
const Size stride = scale.x * 4;
const Size size = stride * scale.y;
const Int_32 fd = CreateShmFile(size);
if (mmap(nullptr, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0) == MAP_FAILED)
{
EHS_LOG_INT(LogType::ERR, 0, "Failed to map shared memory file.");
close(fd);
return;
}
wl_shm_pool *pool = wl_shm_create_pool(shm, fd, (int32_t)size);
buffer = wl_shm_pool_create_buffer(pool, 0, (int32_t)scale.x, (int32_t)scale.y, (int32_t)stride,
WL_SHM_FORMAT_XRGB8888);
wl_shm_pool_destroy(pool);
close(fd);
}
void Window::SurfaceConfigure(void* data, xdg_surface* xdg_surface, UInt_32 serial) void Window::SurfaceConfigure(void* data, xdg_surface* xdg_surface, UInt_32 serial)
{ {
xdg_surface_ack_configure(xdg_surface, serial); xdg_surface_ack_configure(xdg_surface, serial);
@ -16,41 +83,139 @@ namespace ehs
{ {
Serializer<UInt_64>* ser = (Serializer<UInt_64>*)data; Serializer<UInt_64>* ser = (Serializer<UInt_64>*)data;
if (Str_8::Cmp(interface, "wl_compositor")) if (Str_8::Cmp(interface, wl_compositor_interface.name))
{ {
ser->SetOffset(0); ser->SetOffset(0);
wl_compositor** comp = ser->Read<wl_compositor**>(); wl_compositor** comp = ser->Read<wl_compositor**>();
*comp = (wl_compositor*)wl_registry_bind(registry, id, &wl_compositor_interface, 1); *comp = (wl_compositor*)wl_registry_bind(registry, id, &wl_compositor_interface, 1);
} }
else if (Str_8::Cmp(interface, xdg_wm_base_interface.name))
if (Str_8::Cmp(interface, "xdg_wm_base"))
{ {
ser->SetOffset(sizeof(void*)); ser->SetOffset(sizeof(void*));
xdg_wm_base** base = ser->Read<xdg_wm_base**>(); xdg_wm_base** base = ser->Read<xdg_wm_base**>();
*base = (xdg_wm_base*)wl_registry_bind(registry, id, &xdg_wm_base_interface, 1); *base = (xdg_wm_base*)wl_registry_bind(registry, id, &xdg_wm_base_interface, 1);
} }
else if (Str_8::Cmp(interface, wl_shm_interface.name))
{
ser->SetOffset(sizeof(void*) * 2);
wl_shm** shm = ser->Read<wl_shm**>();
*shm = (wl_shm*)wl_registry_bind(registry, id, &wl_shm_interface, 1);
}
else if (Str_8::Cmp(interface, zxdg_decoration_manager_v1_interface.name))
{
ser->SetOffset(sizeof(void*) * 3);
zxdg_decoration_manager_v1** decManager = ser->Read<zxdg_decoration_manager_v1**>();
*decManager = (zxdg_decoration_manager_v1*)wl_registry_bind(registry, id, &zxdg_decoration_manager_v1_interface, 1);
}
} }
void Window::RegistryRemoved(void* data, wl_registry* registry, UInt_32 id) void Window::RegistryRemoved(void* data, wl_registry* registry, UInt_32 id)
{ {
} }
void Window::Resolution(void *data, struct xdg_toplevel *xdg_toplevel, Int_32 width, Int_32 height, wl_array *states)
{
Console::Write_8("Resolution: " + Str_8::FromNum(width) + "x" + Str_8::FromNum(height));
}
Window::~Window() Window::~Window()
{ {
xdg_toplevel_destroy(xdgToplevel); xdg_toplevel_destroy(xdgToplevel);
xdg_surface_destroy(xdgSurface); xdg_surface_destroy(xdgSurface);
xdg_wm_base_destroy(xdgShell); xdg_wm_base_destroy(xdgShell);
wl_surface_destroy(surface); wl_surface_destroy(wlSurface);
wl_compositor_destroy(compositor); wl_compositor_destroy(compositor);
wl_registry_destroy(registry); wl_registry_destroy(registry);
wl_display_disconnect(display); wl_display_disconnect(display);
} }
Window::Window() Window::Window()
: display(nullptr), registry(nullptr), compositor(nullptr) , surface(nullptr) : display(nullptr), registry(nullptr), compositor(nullptr), wlSurface(nullptr), xdgShell(nullptr),
xdgSurface(nullptr), xdgToplevel(nullptr), shm(nullptr), buffer(nullptr), decManager(nullptr), dec(nullptr)
{ {
} }
Window::Window(Window &&win) noexcept
: BaseWindow(win), display(win.display), registry(win.registry), compositor(win.compositor),
wlSurface(win.wlSurface), xdgShell(win.xdgShell), xdgSurface(win.xdgSurface), xdgToplevel(win.xdgToplevel),
shm(win.shm), buffer(win.buffer), decManager(win.decManager), dec(win.dec)
{
win.display = nullptr;
win.registry = nullptr;
win.compositor = nullptr;
win.wlSurface = nullptr;
win.xdgShell = nullptr;
win.xdgSurface = nullptr;
win.xdgToplevel = nullptr;
win.shm = nullptr;
win.buffer = nullptr;
win.decManager = nullptr;
win.dec = nullptr;
}
Window::Window(const Window &win)
: BaseWindow(win), display(nullptr), registry(nullptr), compositor(nullptr), wlSurface(nullptr),
xdgShell(nullptr), xdgSurface(nullptr), xdgToplevel(nullptr), shm(nullptr), buffer(nullptr), decManager(nullptr),
dec(nullptr)
{
}
Window & Window::operator=(Window &&win) noexcept
{
if (this == &win)
return *this;
BaseWindow::operator=(win);
display = win.display;
registry = win.registry;
compositor = win.compositor;
wlSurface = win.wlSurface;
xdgShell = win.xdgShell;
xdgSurface = win.xdgSurface;
xdgToplevel = win.xdgToplevel;
shm = win.shm;
buffer = win.buffer;
decManager = win.decManager;
dec = win.dec;
win.display = nullptr;
win.registry = nullptr;
win.compositor = nullptr;
win.wlSurface = nullptr;
win.xdgShell = nullptr;
win.xdgSurface = nullptr;
win.xdgToplevel = nullptr;
win.shm = nullptr;
win.buffer = nullptr;
win.decManager = nullptr;
win.dec = nullptr;
return *this;
}
Window & Window::operator=(const Window &win)
{
if (this == &win)
return *this;
BaseWindow::operator=(win);
display = nullptr;
registry = nullptr;
compositor = nullptr;
wlSurface = nullptr;
xdgShell = nullptr;
xdgSurface = nullptr;
xdgToplevel = nullptr;
shm = nullptr;
buffer = nullptr;
decManager = nullptr;
dec = nullptr;
return *this;
}
void Window::Create_32(const Str_32& title, const Vec2_s32& pos, const Vec2_u32 scale) void Window::Create_32(const Str_32& title, const Vec2_s32& pos, const Vec2_u32 scale)
{ {
Create_8(UTF::To_8(title), pos, scale); Create_8(UTF::To_8(title), pos, scale);
@ -73,6 +238,8 @@ namespace ehs
Serializer<UInt_64> data(Endianness::LE); Serializer<UInt_64> data(Endianness::LE);
data.Write(&compositor); data.Write(&compositor);
data.Write(&xdgShell); data.Write(&xdgShell);
data.Write(&shm);
data.Write(&decManager);
static constexpr wl_registry_listener registry_listener = { static constexpr wl_registry_listener registry_listener = {
RegistryHandler, RegistryHandler,
@ -95,22 +262,41 @@ namespace ehs
xdg_wm_base_add_listener(xdgShell, &xdg_shell_listener, nullptr); xdg_wm_base_add_listener(xdgShell, &xdg_shell_listener, nullptr);
surface = wl_compositor_create_surface(compositor); wlSurface = wl_compositor_create_surface(compositor);
if (!surface) if (!wlSurface)
{ {
EHS_LOG_INT(LogType::ERR, 2, "Can't create surface."); EHS_LOG_INT(LogType::ERR, 2, "Can't create surface.");
return; return;
} }
xdgSurface = xdg_wm_base_get_xdg_surface(xdgShell, surface); CreateBuffer(scale);
xdgToplevel = xdg_surface_get_toplevel(xdgSurface);
xdgSurface = xdg_wm_base_get_xdg_surface(xdgShell, wlSurface);
static constexpr xdg_surface_listener surfaceListener = {SurfaceConfigure}; static constexpr xdg_surface_listener surfaceListener = {SurfaceConfigure};
xdg_surface_add_listener(xdgSurface, &surfaceListener, nullptr); xdg_surface_add_listener(xdgSurface, &surfaceListener, nullptr);
xdgToplevel = xdg_surface_get_toplevel(xdgSurface);
static constexpr xdg_toplevel_listener topLevelListener = {
.configure = Resolution
};
xdg_toplevel_add_listener(xdgToplevel, &topLevelListener, nullptr);
xdg_toplevel_set_title(xdgToplevel, title); xdg_toplevel_set_title(xdgToplevel, title);
wl_surface_commit(surface); xdg_toplevel_set_app_id(xdgToplevel, "dsadasfsaAS");
dec = zxdg_decoration_manager_v1_get_toplevel_decoration(decManager, xdgToplevel);
zxdg_toplevel_decoration_v1_set_mode(dec, ZXDG_TOPLEVEL_DECORATION_V1_MODE_SERVER_SIDE);
wl_surface_attach(wlSurface, buffer, 0, 0);
wl_surface_commit(wlSurface);
wl_display_roundtrip(display);
OnCreated();
} }
void Window::OnCreated() void Window::OnCreated()
@ -128,8 +314,8 @@ namespace ehs
xdg_wm_base_destroy(xdgShell); xdg_wm_base_destroy(xdgShell);
xdgShell = nullptr; xdgShell = nullptr;
wl_surface_destroy(surface); wl_surface_destroy(wlSurface);
surface = nullptr; wlSurface = nullptr;
wl_compositor_destroy(compositor); wl_compositor_destroy(compositor);
compositor = nullptr; compositor = nullptr;