2021-10-19 16:46:46 -04:00
|
|
|
// somebar - dwl bar
|
|
|
|
// See LICENSE file for copyright and license details.
|
|
|
|
|
2021-10-25 09:49:01 -04:00
|
|
|
#include <cstdio>
|
2021-10-22 10:32:53 -04:00
|
|
|
#include <fcntl.h>
|
2021-10-19 16:46:46 -04:00
|
|
|
#include <math.h>
|
2021-10-22 10:32:53 -04:00
|
|
|
#include <signal.h>
|
2021-10-19 16:46:46 -04:00
|
|
|
#include <sys/mman.h>
|
2021-10-22 10:39:19 -04:00
|
|
|
#include <sys/signalfd.h>
|
2021-10-22 10:32:53 -04:00
|
|
|
#include <sys/stat.h>
|
2021-10-19 16:46:46 -04:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
2021-10-22 11:26:05 -04:00
|
|
|
#include <linux/input-event-codes.h>
|
2021-10-22 10:52:04 -04:00
|
|
|
#include <optional>
|
2021-10-22 09:34:19 -04:00
|
|
|
#include <QGuiApplication>
|
2021-10-19 16:46:46 -04:00
|
|
|
#include <QSocketNotifier>
|
|
|
|
#include <wayland-client.h>
|
2021-10-22 11:34:21 -04:00
|
|
|
#include <wayland-cursor.h>
|
2021-10-19 16:46:46 -04:00
|
|
|
#include "wlr-layer-shell-unstable-v1-client-protocol.h"
|
|
|
|
#include "xdg-shell-client-protocol.h"
|
2021-10-24 13:04:29 -04:00
|
|
|
#include "net-tapesoftware-dwl-wm-unstable-v1-client-protocol.h"
|
2021-10-20 14:20:27 -04:00
|
|
|
#include "common.hpp"
|
2021-10-20 14:45:23 -04:00
|
|
|
#include "bar.hpp"
|
2021-10-19 16:46:46 -04:00
|
|
|
|
2021-10-25 09:49:01 -04:00
|
|
|
struct Monitor {
|
|
|
|
uint32_t name;
|
|
|
|
wl_output *wlOutput;
|
|
|
|
znet_tapesoftware_dwl_wm_monitor_v1 *dwlMonitor;
|
|
|
|
std::optional<Bar> bar;
|
|
|
|
};
|
|
|
|
|
2021-10-22 09:05:47 -04:00
|
|
|
static void waylandFlush();
|
2021-10-19 16:46:46 -04:00
|
|
|
static void waylandWriteReady();
|
|
|
|
static void requireGlobal(const void *p, const char *name);
|
2021-10-22 10:32:53 -04:00
|
|
|
static void setupStatusFifo();
|
|
|
|
static void onStatus();
|
|
|
|
static void cleanup();
|
2021-10-19 16:46:46 -04:00
|
|
|
|
2021-10-20 14:20:27 -04:00
|
|
|
wl_display *display;
|
|
|
|
wl_compositor *compositor;
|
|
|
|
wl_shm *shm;
|
|
|
|
zwlr_layer_shell_v1 *wlrLayerShell;
|
2021-10-24 13:04:29 -04:00
|
|
|
znet_tapesoftware_dwl_wm_v1 *dwlWm;
|
|
|
|
std::vector<QString> tagNames;
|
2021-10-25 09:49:01 -04:00
|
|
|
static std::vector<Monitor> monitors;
|
2021-10-22 10:52:04 -04:00
|
|
|
static std::optional<Bar> bar;
|
2021-10-22 10:32:53 -04:00
|
|
|
static std::string statusFifoName;
|
|
|
|
static int statusFifoFd {-1};
|
|
|
|
static int statusFifoWriter {-1};
|
2021-10-19 16:46:46 -04:00
|
|
|
static QSocketNotifier *displayWriteNotifier;
|
2021-10-22 10:39:19 -04:00
|
|
|
static bool quitting {false};
|
2021-10-19 16:46:46 -04:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-10-22 11:26:05 -04:00
|
|
|
struct PointerState {
|
|
|
|
wl_pointer *pointer;
|
2021-10-22 11:34:21 -04:00
|
|
|
wl_surface *cursorSurface;
|
|
|
|
wl_cursor_image *cursorImage;
|
2021-10-22 11:26:05 -04:00
|
|
|
Bar *focusedBar;
|
|
|
|
int x, y;
|
|
|
|
bool leftButtonClick;
|
|
|
|
};
|
|
|
|
static PointerState pointerState;
|
|
|
|
static const struct wl_pointer_listener pointerListener = {
|
2021-10-22 11:34:21 -04:00
|
|
|
.enter = [](void*, wl_pointer *pointer, uint32_t serial,
|
|
|
|
wl_surface*, wl_fixed_t x, wl_fixed_t y)
|
2021-10-22 11:26:05 -04:00
|
|
|
{
|
|
|
|
pointerState.focusedBar = &bar.value();
|
2021-10-22 11:34:21 -04:00
|
|
|
wl_pointer_set_cursor(pointer, serial, pointerState.cursorSurface,
|
|
|
|
pointerState.cursorImage->hotspot_x, pointerState.cursorImage->hotspot_y);
|
2021-10-22 11:26:05 -04:00
|
|
|
},
|
|
|
|
.leave = [](void*, wl_pointer*, uint32_t serial, wl_surface*) {
|
|
|
|
pointerState.focusedBar = nullptr;
|
|
|
|
},
|
|
|
|
.motion = [](void*, wl_pointer*, uint32_t, wl_fixed_t x, wl_fixed_t y) {
|
|
|
|
pointerState.x = wl_fixed_to_int(x);
|
|
|
|
pointerState.y = wl_fixed_to_int(y);
|
|
|
|
},
|
|
|
|
.button = [](void*, wl_pointer*, uint32_t, uint32_t, uint32_t button, uint32_t pressed) {
|
|
|
|
if (button == BTN_LEFT) {
|
|
|
|
pointerState.leftButtonClick = pressed == WL_POINTER_BUTTON_STATE_PRESSED;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.axis = [](void*, wl_pointer*, uint32_t, uint32_t, wl_fixed_t) { },
|
|
|
|
.frame = [](void*, wl_pointer*) {
|
|
|
|
if (!pointerState.focusedBar) return;
|
|
|
|
if (pointerState.leftButtonClick) {
|
|
|
|
pointerState.leftButtonClick = false;
|
|
|
|
pointerState.focusedBar->click(pointerState.x, pointerState.y);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.axis_source = [](void*, wl_pointer*, uint32_t) { },
|
|
|
|
.axis_stop = [](void*, wl_pointer*, uint32_t, uint32_t) { },
|
|
|
|
.axis_discrete = [](void*, wl_pointer*, uint32_t, int32_t) { },
|
|
|
|
};
|
|
|
|
|
|
|
|
static wl_seat *seat;
|
|
|
|
static const struct wl_seat_listener seatListener = {
|
|
|
|
[](void*, wl_seat*, uint32_t cap)
|
|
|
|
{
|
2021-10-22 11:34:21 -04:00
|
|
|
if (!pointerState.pointer && WL_SEAT_CAPABILITY_POINTER) {
|
|
|
|
auto cursorTheme = wl_cursor_theme_load(NULL, 24, shm);
|
|
|
|
auto cursorImage = wl_cursor_theme_get_cursor(cursorTheme, "left_ptr")->images[0];
|
|
|
|
pointerState.cursorImage = cursorImage;
|
|
|
|
pointerState.cursorSurface = wl_compositor_create_surface(compositor);
|
|
|
|
wl_surface_attach(pointerState.cursorSurface,
|
|
|
|
wl_cursor_image_get_buffer(cursorImage), 0, 0);
|
|
|
|
wl_surface_commit(pointerState.cursorSurface);
|
2021-10-22 11:26:05 -04:00
|
|
|
pointerState.pointer = wl_seat_get_pointer(seat);
|
|
|
|
wl_pointer_add_listener(pointerState.pointer, &pointerListener, nullptr);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
[](void*, wl_seat*, const char *name) { }
|
|
|
|
};
|
|
|
|
|
2021-10-24 13:04:29 -04:00
|
|
|
static const struct znet_tapesoftware_dwl_wm_v1_listener dwlWmListener = {
|
|
|
|
.tag = [](void*, znet_tapesoftware_dwl_wm_v1*, const char *name) {
|
|
|
|
tagNames.push_back(name);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2021-10-25 09:49:01 -04:00
|
|
|
static const struct znet_tapesoftware_dwl_wm_monitor_v1_listener dwlWmMonitorListener {
|
|
|
|
.tag = [](void*, znet_tapesoftware_dwl_wm_monitor_v1*, int tag, int active, int numClients) {
|
|
|
|
printf("tag %s: active=%d, num_clients=%d\n", qPrintable(tagNames[tag]), active, numClients);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static void setupOutput(Monitor &monitor) {
|
|
|
|
monitor.dwlMonitor = znet_tapesoftware_dwl_wm_v1_get_monitor(dwlWm, monitor.wlOutput);
|
|
|
|
znet_tapesoftware_dwl_wm_monitor_v1_add_listener(monitor.dwlMonitor, &dwlWmMonitorListener, &monitor);
|
|
|
|
monitor.bar.emplace(monitor.wlOutput);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void onOutput(int name, wl_output *output) {
|
|
|
|
auto& monitor = monitors.emplace_back(name, output);
|
|
|
|
if (dwlWm) {
|
|
|
|
setupOutput(monitor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-19 16:46:46 -04:00
|
|
|
// called after we have received the initial batch of globals
|
|
|
|
static void onReady()
|
|
|
|
{
|
|
|
|
requireGlobal(compositor, "wl_compositor");
|
|
|
|
requireGlobal(shm, "wl_shm");
|
2021-10-22 11:26:05 -04:00
|
|
|
requireGlobal(seat, "wl_seat");
|
2021-10-19 16:46:46 -04:00
|
|
|
requireGlobal(wlrLayerShell, "zwlr_layer_shell_v1");
|
2021-10-25 09:49:01 -04:00
|
|
|
requireGlobal(dwlWm, "znet_tapesoftware_dwl_wm_v1");
|
2021-10-22 10:32:53 -04:00
|
|
|
setupStatusFifo();
|
2021-10-25 09:49:01 -04:00
|
|
|
wl_display_roundtrip(display); // roundtrip so we receive all dwl tags etc.
|
|
|
|
for (auto& monitor : monitors) {
|
|
|
|
auto monitor = znet_tapesoftware_dwl_wm_v1_get_monitor(dwlWm, output);
|
|
|
|
printf("created monitor %p for output %p\n", monitor, output);
|
|
|
|
bar.emplace(output);
|
|
|
|
}
|
2021-10-19 16:46:46 -04:00
|
|
|
}
|
|
|
|
|
2021-10-22 10:32:53 -04:00
|
|
|
static void setupStatusFifo()
|
|
|
|
{
|
|
|
|
for (auto i=0; i<100; i++) {
|
|
|
|
auto path = std::string{getenv("XDG_RUNTIME_DIR")} + "/somebar-" + std::to_string(i);
|
|
|
|
auto result = mkfifo(path.c_str(), 0666);
|
|
|
|
if (result == 0) {
|
|
|
|
auto fd = open(path.c_str(), O_CLOEXEC | O_NONBLOCK | O_RDONLY);
|
|
|
|
if (fd == -1) {
|
|
|
|
perror("open status fifo reader");
|
|
|
|
cleanup();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
statusFifoName = path;
|
|
|
|
statusFifoFd = fd;
|
|
|
|
|
|
|
|
fd = open(path.c_str(), O_CLOEXEC | O_WRONLY);
|
|
|
|
if (fd == -1) {
|
|
|
|
perror("open status fifo writer");
|
|
|
|
cleanup();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
statusFifoWriter = fd;
|
|
|
|
|
|
|
|
auto statusNotifier = new QSocketNotifier(statusFifoFd, QSocketNotifier::Read);
|
|
|
|
statusNotifier->setEnabled(true);
|
|
|
|
QObject::connect(statusNotifier, &QSocketNotifier::activated, onStatus);
|
|
|
|
return;
|
|
|
|
} else if (errno != EEXIST) {
|
|
|
|
perror("mkfifo");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void onStatus()
|
|
|
|
{
|
|
|
|
char buffer[512];
|
|
|
|
auto n = read(statusFifoFd, buffer, sizeof(buffer));
|
|
|
|
auto str = QString::fromUtf8(buffer, n);
|
2021-10-22 10:52:04 -04:00
|
|
|
if (bar) {
|
|
|
|
bar->setStatus(str);
|
|
|
|
}
|
2021-10-22 10:32:53 -04:00
|
|
|
}
|
|
|
|
|
2021-10-19 16:46:46 -04:00
|
|
|
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;
|
|
|
|
}
|
2021-10-22 11:26:05 -04:00
|
|
|
if (seat == nullptr && reg.handle(seat, wl_seat_interface, 7)) {
|
|
|
|
wl_seat_add_listener(seat, &seatListener, nullptr);
|
|
|
|
}
|
2021-10-25 09:49:01 -04:00
|
|
|
if (wl_output *output; reg.handle(output, wl_output_interface, 1)) {
|
|
|
|
outputs.push_back(output);
|
|
|
|
}
|
2021-10-24 13:04:29 -04:00
|
|
|
if (reg.handle(dwlWm, znet_tapesoftware_dwl_wm_v1_interface, 1)) {
|
|
|
|
znet_tapesoftware_dwl_wm_v1_add_listener(dwlWm, &dwlWmListener, nullptr);
|
|
|
|
}
|
2021-10-19 16:46:46 -04:00
|
|
|
}
|
|
|
|
static const struct wl_registry_listener registry_listener = { registryHandleGlobal, nullptr };
|
|
|
|
|
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2021-10-22 10:39:19 -04:00
|
|
|
static sigset_t blockedsigs;
|
|
|
|
sigemptyset(&blockedsigs);
|
|
|
|
sigaddset(&blockedsigs, SIGINT);
|
|
|
|
sigaddset(&blockedsigs, SIGTERM);
|
|
|
|
sigprocmask(SIG_BLOCK, &blockedsigs, nullptr);
|
|
|
|
|
2021-10-22 09:34:19 -04:00
|
|
|
QGuiApplication app(argc, argv);
|
2021-10-19 16:46:46 -04:00
|
|
|
QCoreApplication::setOrganizationName("tape software");
|
|
|
|
QCoreApplication::setOrganizationDomain("tapesoftware.net");
|
|
|
|
QCoreApplication::setApplicationName("somebar");
|
|
|
|
|
2021-10-22 10:39:19 -04:00
|
|
|
int sfd = signalfd(-1, &blockedsigs, SFD_CLOEXEC | SFD_NONBLOCK);
|
|
|
|
if (sfd < 0) {
|
|
|
|
perror("signalfd");
|
|
|
|
cleanup();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
QSocketNotifier signalNotifier {sfd, QSocketNotifier::Read};
|
|
|
|
QObject::connect(&signalNotifier, &QSocketNotifier::activated, []() { quitting = true; });
|
2021-10-22 10:32:53 -04:00
|
|
|
|
2021-10-19 16:46:46 -04:00
|
|
|
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);
|
2021-10-22 10:32:53 -04:00
|
|
|
wl_display_roundtrip(display);
|
|
|
|
onReady();
|
2021-10-19 16:46:46 -04:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2021-10-22 10:32:53 -04:00
|
|
|
while (!quitting) {
|
2021-10-22 09:05:47 -04:00
|
|
|
waylandFlush();
|
|
|
|
app.processEvents(QEventLoop::WaitForMoreEvents);
|
|
|
|
}
|
2021-10-22 10:32:53 -04:00
|
|
|
cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cleanup() {
|
|
|
|
if (!statusFifoName.empty()) {
|
|
|
|
unlink(statusFifoName.c_str());
|
|
|
|
}
|
2021-10-19 16:46:46 -04:00
|
|
|
}
|
|
|
|
|
2021-10-20 14:20:27 -04:00
|
|
|
void waylandFlush()
|
2021-10-19 16:46:46 -04:00
|
|
|
{
|
|
|
|
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);
|
2021-10-22 10:32:53 -04:00
|
|
|
cleanup();
|
2021-10-19 16:46:46 -04:00
|
|
|
exit(1);
|
|
|
|
}
|