2018-12-13 18:03:44 +01:00
|
|
|
#include <errno.h>
|
2018-12-21 19:00:32 +01:00
|
|
|
#include <stdbool.h>
|
2018-12-12 22:29:57 +01:00
|
|
|
#include <stdio.h>
|
2018-12-13 18:03:44 +01:00
|
|
|
#include <stdlib.h>
|
2018-12-21 19:00:32 +01:00
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <unistd.h>
|
2018-12-12 22:29:57 +01:00
|
|
|
#include "xdpw.h"
|
|
|
|
|
|
|
|
static const char object_path[] = "/org/freedesktop/portal/desktop";
|
|
|
|
static const char interface_name[] = "org.freedesktop.impl.portal.Screenshot";
|
|
|
|
|
2018-12-21 19:00:32 +01:00
|
|
|
static bool exec_screenshooter(const char *path) {
|
|
|
|
pid_t pid = fork();
|
|
|
|
if (pid < 0) {
|
|
|
|
perror("fork");
|
|
|
|
return false;
|
|
|
|
} else if (pid == 0) {
|
|
|
|
char *const argv[] = {
|
|
|
|
"grim",
|
|
|
|
"--",
|
|
|
|
(char *)path,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
execvp("grim", argv);
|
|
|
|
|
|
|
|
perror("execvp");
|
|
|
|
exit(127);
|
|
|
|
}
|
|
|
|
|
|
|
|
int stat;
|
|
|
|
if (waitpid(pid, &stat, 0) < 0) {
|
|
|
|
perror("waitpid");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return stat == 0;
|
|
|
|
}
|
|
|
|
|
2018-12-12 22:29:57 +01:00
|
|
|
static int method_screenshot(sd_bus_message *msg, void *data,
|
|
|
|
sd_bus_error *ret_error) {
|
2018-12-13 18:03:44 +01:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
char *handle, *app_id, *parent_window;
|
|
|
|
ret = sd_bus_message_read(msg, "oss", &handle, &app_id, &parent_window);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
// TODO: read options
|
|
|
|
|
|
|
|
// TODO: cleanup this
|
|
|
|
struct xdpw_request *req =
|
2020-04-16 10:21:55 +02:00
|
|
|
xdpw_request_create(sd_bus_message_get_bus(msg), handle);
|
2018-12-13 18:03:44 +01:00
|
|
|
if (req == NULL) {
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
|
2018-12-21 19:00:32 +01:00
|
|
|
// TODO: choose a better path
|
|
|
|
const char path[] = "/tmp/out.png";
|
|
|
|
if (!exec_screenshooter(path)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char uri_prefix[] = "file://";
|
|
|
|
char uri[strlen(path) + strlen(uri_prefix) + 1];
|
|
|
|
snprintf(uri, sizeof(uri), "%s%s", uri_prefix, path);
|
|
|
|
|
2019-07-24 16:59:40 +02:00
|
|
|
sd_bus_message *reply = NULL;
|
|
|
|
ret = sd_bus_message_new_method_return(msg, &reply);
|
2018-12-13 18:03:44 +01:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-07-24 16:59:40 +02:00
|
|
|
ret = sd_bus_message_append(reply, "ua{sv}", PORTAL_RESPONSE_SUCCESS, 1, "uri", "s", uri);
|
2018-12-13 18:03:44 +01:00
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = sd_bus_send(NULL, reply, NULL);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
sd_bus_message_unref(reply);
|
2018-12-12 22:29:57 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const sd_bus_vtable screenshot_vtable[] = {
|
|
|
|
SD_BUS_VTABLE_START(0),
|
|
|
|
SD_BUS_METHOD("Screenshot", "ossa{sv}", "ua{sv}", method_screenshot, SD_BUS_VTABLE_UNPRIVILEGED),
|
|
|
|
SD_BUS_VTABLE_END
|
|
|
|
};
|
|
|
|
|
2020-04-16 10:21:55 +02:00
|
|
|
int xdpw_screenshot_init(struct xdpw_state *state) {
|
2018-12-12 22:29:57 +01:00
|
|
|
// TODO: cleanup
|
|
|
|
sd_bus_slot *slot = NULL;
|
2020-03-10 20:46:37 +01:00
|
|
|
return sd_bus_add_object_vtable(state->bus, &slot, object_path, interface_name,
|
2018-12-12 22:29:57 +01:00
|
|
|
screenshot_vtable, NULL);
|
|
|
|
}
|