177 lines
6.2 KiB
C++
177 lines
6.2 KiB
C++
|
// somebar - dwl bar
|
||
|
// See LICENSE file for copyright and license details.
|
||
|
|
||
|
#include <math.h>
|
||
|
#include <sys/mman.h>
|
||
|
#include <sys/types.h>
|
||
|
#include <unistd.h>
|
||
|
#include <QCoreApplication>
|
||
|
#include <QSocketNotifier>
|
||
|
#include <wayland-client.h>
|
||
|
#include "qnamespace.h"
|
||
|
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
||
|
#include "xdg-shell-client-protocol.h"
|
||
|
|
||
|
constexpr uint32_t barSize = 20;
|
||
|
|
||
|
static void waylandFlush();
|
||
|
static void waylandWriteReady();
|
||
|
static void requireGlobal(const void *p, const char *name);
|
||
|
|
||
|
// wayland globals
|
||
|
static wl_display *display;
|
||
|
static wl_compositor *compositor;
|
||
|
static wl_shm *shm;
|
||
|
static zwlr_layer_shell_v1 *wlrLayerShell;
|
||
|
static QSocketNotifier *displayWriteNotifier;
|
||
|
|
||
|
static xdg_wm_base *xdgWmBase;
|
||
|
static const struct xdg_wm_base_listener xdgWmBaseListener = {
|
||
|
[](void*, xdg_wm_base *sender, uint32_t serial) {
|
||
|
xdg_wm_base_pong(sender, serial);
|
||
|
waylandFlush();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
static const struct wl_buffer_listener wlBufferListener {
|
||
|
[](void*, wl_buffer *buffer) {
|
||
|
wl_buffer_destroy(buffer);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// app globals
|
||
|
static wl_surface *surface;
|
||
|
static zwlr_layer_surface_v1 *layerSurface;
|
||
|
static const struct wl_surface_listener surfaceListener = {
|
||
|
// todo
|
||
|
};
|
||
|
static const struct zwlr_layer_surface_v1_listener layerSurfaceListener = {
|
||
|
[](void*, zwlr_layer_surface_v1 *layerSurface, uint32_t serial, uint32_t width, uint32_t height) {
|
||
|
zwlr_layer_surface_v1_ack_configure(layerSurface, serial);
|
||
|
printf("configured to %d x %d\n", width, height);
|
||
|
auto stride = width * 4;
|
||
|
auto size = stride * height;
|
||
|
auto fd = memfd_create("somebar-surface", MFD_CLOEXEC);
|
||
|
ftruncate(fd, size);
|
||
|
auto buffer = reinterpret_cast<char*>(mmap(nullptr, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0));
|
||
|
auto pool = wl_shm_create_pool(shm, fd, size);
|
||
|
auto wlBuffer = wl_shm_pool_create_buffer(pool, 0, width, height, stride, WL_SHM_FORMAT_XRGB8888);
|
||
|
wl_shm_pool_destroy(pool);
|
||
|
close(fd);
|
||
|
|
||
|
auto w = 2*M_PI/(width / 10);
|
||
|
for (auto x = 0; x < width; x++) {
|
||
|
auto val = 255*(sin(x*w)/2+0.5);
|
||
|
for (auto y = 0; y < height; y++) {
|
||
|
auto p = &buffer[y*stride+x*4];
|
||
|
*p++ = 0;
|
||
|
*p++ = 0;
|
||
|
*p++ = val;
|
||
|
*p++ = val;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
wl_buffer_add_listener(wlBuffer, &wlBufferListener, nullptr);
|
||
|
wl_surface_attach(surface, wlBuffer, 0, 0);
|
||
|
wl_surface_commit(surface);
|
||
|
waylandFlush();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// called after we have received the initial batch of globals
|
||
|
static void onReady()
|
||
|
{
|
||
|
requireGlobal(compositor, "wl_compositor");
|
||
|
requireGlobal(shm, "wl_shm");
|
||
|
requireGlobal(wlrLayerShell, "zwlr_layer_shell_v1");
|
||
|
surface = wl_compositor_create_surface(compositor);
|
||
|
layerSurface = zwlr_layer_shell_v1_get_layer_surface(wlrLayerShell, surface, nullptr, ZWLR_LAYER_SHELL_V1_LAYER_TOP, "net.tapesoftware.Somebar");
|
||
|
zwlr_layer_surface_v1_add_listener(layerSurface, &layerSurfaceListener, nullptr);
|
||
|
zwlr_layer_surface_v1_set_anchor(layerSurface, ZWLR_LAYER_SURFACE_V1_ANCHOR_TOP | ZWLR_LAYER_SURFACE_V1_ANCHOR_LEFT | ZWLR_LAYER_SURFACE_V1_ANCHOR_RIGHT);
|
||
|
zwlr_layer_surface_v1_set_size(layerSurface, 0, barSize);
|
||
|
zwlr_layer_surface_v1_set_exclusive_zone(layerSurface, barSize);
|
||
|
wl_surface_commit(surface);
|
||
|
waylandFlush();
|
||
|
}
|
||
|
|
||
|
struct HandleGlobalHelper {
|
||
|
wl_registry *registry;
|
||
|
uint32_t name;
|
||
|
const char *interface;
|
||
|
|
||
|
template<typename T>
|
||
|
bool handle(T &store, const wl_interface &iface, int version) {
|
||
|
if (strcmp(interface, iface.name)) return false;
|
||
|
store = static_cast<T>(wl_registry_bind(registry, name, &iface, version));
|
||
|
return true;
|
||
|
}
|
||
|
};
|
||
|
static void registryHandleGlobal(void*, wl_registry *registry, uint32_t name, const char *interface, uint32_t version)
|
||
|
{
|
||
|
auto reg = HandleGlobalHelper { registry, name, interface };
|
||
|
printf("got global: %s v%d\n", interface, version);
|
||
|
if (reg.handle(compositor, wl_compositor_interface, 4)) return;
|
||
|
if (reg.handle(shm, wl_shm_interface, 1)) return;
|
||
|
if (reg.handle(wlrLayerShell, zwlr_layer_shell_v1_interface, 4)) return;
|
||
|
if (reg.handle(xdgWmBase, xdg_wm_base_interface, 2)) {
|
||
|
xdg_wm_base_add_listener(xdgWmBase, &xdgWmBaseListener, nullptr);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
static const struct wl_registry_listener registry_listener = { registryHandleGlobal, nullptr };
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
QCoreApplication app(argc, argv);
|
||
|
QCoreApplication::setOrganizationName("tape software");
|
||
|
QCoreApplication::setOrganizationDomain("tapesoftware.net");
|
||
|
QCoreApplication::setApplicationName("somebar");
|
||
|
|
||
|
display = wl_display_connect(NULL);
|
||
|
if (!display) {
|
||
|
fprintf(stderr, "Failed to connect to Wayland display\n");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
auto registry = wl_display_get_registry(display);
|
||
|
wl_registry_add_listener(registry, ®istry_listener, nullptr);
|
||
|
|
||
|
auto initialSync = wl_display_sync(display);
|
||
|
auto initialSyncListener = wl_callback_listener {
|
||
|
[](void*, wl_callback *cb, uint32_t) {
|
||
|
onReady();
|
||
|
wl_callback_destroy(cb);
|
||
|
}
|
||
|
};
|
||
|
wl_callback_add_listener(initialSync, &initialSyncListener, nullptr);
|
||
|
|
||
|
QSocketNotifier displayReadNotifier(wl_display_get_fd(display), QSocketNotifier::Read);
|
||
|
displayReadNotifier.setEnabled(true);
|
||
|
QObject::connect(&displayReadNotifier, &QSocketNotifier::activated, [=]() { wl_display_dispatch(display); });
|
||
|
displayWriteNotifier = new QSocketNotifier(wl_display_get_fd(display), QSocketNotifier::Write);
|
||
|
displayWriteNotifier->setEnabled(false);
|
||
|
QObject::connect(displayWriteNotifier, &QSocketNotifier::activated, waylandWriteReady);
|
||
|
waylandFlush();
|
||
|
|
||
|
return app.exec();
|
||
|
}
|
||
|
|
||
|
static void waylandFlush()
|
||
|
{
|
||
|
wl_display_dispatch_pending(display);
|
||
|
if (wl_display_flush(display) < 0 && errno == EAGAIN) {
|
||
|
displayWriteNotifier->setEnabled(true);
|
||
|
}
|
||
|
}
|
||
|
static void waylandWriteReady()
|
||
|
{
|
||
|
displayWriteNotifier->setEnabled(false);
|
||
|
waylandFlush();
|
||
|
}
|
||
|
static void requireGlobal(const void *p, const char *name)
|
||
|
{
|
||
|
if (p) return;
|
||
|
fprintf(stderr, "Wayland compositor does not export required global %s, aborting.\n", name);
|
||
|
exit(1);
|
||
|
}
|