diff --git a/cfg/checks/x11.mk b/cfg/checks/x11.mk
index ba3683156..b5b2ed94c 100644
--- a/cfg/checks/x11.mk
+++ b/cfg/checks/x11.mk
@@ -1,7 +1,7 @@
# Variables for X11 support
X11_LIBS = x11
X11_CFLAGS = -DX11
-X11_OBJ = xtra.o
+X11_OBJ = x11focus.o
# Check if we can build X11 support
CHECK_X11_LIBS = $(shell $(PKG_CONFIG) --exists $(X11_LIBS) || echo -n "error")
diff --git a/src/notify.c b/src/notify.c
index 629878302..c8b2ed8d9 100644
--- a/src/notify.c
+++ b/src/notify.c
@@ -35,7 +35,7 @@
#include "misc_tools.h"
#include "notify.h"
#include "settings.h"
-#include "xtra.h"
+#include "x11focus.h"
#if defined(AUDIO) || defined(SOUND_NOTIFY)
#ifdef __APPLE__
diff --git a/src/toxic.c b/src/toxic.c
index 92c03215c..0d8852736 100644
--- a/src/toxic.c
+++ b/src/toxic.c
@@ -66,7 +66,7 @@
#include "windows.h"
#ifdef X11
-#include "xtra.h"
+#include "x11focus.h"
#endif
#ifdef AUDIO
@@ -205,10 +205,7 @@ void exit_toxic_success(Tox *m)
curl_global_cleanup();
#ifdef X11
- /* We have to terminate xtra last coz reasons
- * Please don't call this anywhere else coz trust me
- */
- terminate_xtra();
+ terminate_x11focus();
#endif /* X11 */
exit(EXIT_SUCCESS);
@@ -1350,21 +1347,6 @@ static void init_default_data_files(void)
free(user_config_dir);
}
-// this doesn't do anything (yet)
-#ifdef X11
-void DnD_callback(const char *asdv, DropType dt)
-{
- UNUSED_VAR(asdv);
- UNUSED_VAR(dt);
- // if (dt != DT_plain)
- // return;
-
- // pthread_mutex_lock(&Winthread.lock);
- // line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, asdv);
- // pthread_mutex_unlock(&Winthread.lock);
-}
-#endif /* X11 */
-
int main(int argc, char **argv)
{
/* Make sure all written files are read/writeable only by the current user. */
@@ -1422,7 +1404,7 @@ int main(int argc, char **argv)
#ifdef X11
- if (init_xtra(DnD_callback) == -1) {
+ if (init_x11focus() == -1) {
queue_init_message("X failed to initialize");
}
diff --git a/src/x11focus.c b/src/x11focus.c
new file mode 100644
index 000000000..34a1ff3f7
--- /dev/null
+++ b/src/x11focus.c
@@ -0,0 +1,95 @@
+/* x11focus.c
+ *
+ *
+ * Copyright (C) 2020 Toxic All Rights Reserved.
+ *
+ * This file is part of Toxic.
+ *
+ * Toxic is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Toxic is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Toxic. If not, see .
+ *
+ */
+
+#include "x11focus.h"
+
+#ifndef __APPLE__
+
+#include
+
+static struct Focus {
+ Display *display;
+ Window terminal_window;
+} Focus;
+
+static long unsigned int focused_window_id(void)
+{
+ if (!Focus.display) {
+ return 0;
+ }
+
+ Window focus;
+ int revert;
+
+ XGetInputFocus(Focus.display, &focus, &revert);
+
+ return focus;
+}
+
+bool is_focused(void)
+{
+ if (!Focus.display) {
+ return false;
+ }
+
+ XLockDisplay(Focus.display);
+ bool ret = Focus.terminal_window == focused_window_id();
+ XUnlockDisplay(Focus.display);
+
+ return ret;
+}
+
+int init_x11focus(void)
+{
+ if (XInitThreads() == 0) {
+ return -1;
+ }
+
+ Focus.display = XOpenDisplay(NULL);
+
+ if (!Focus.display) {
+ return -1;
+ }
+
+ Focus.terminal_window = focused_window_id();
+
+ return 0;
+}
+
+void terminate_x11focus(void)
+{
+ if (!Focus.display) {
+ return;
+ }
+
+ XLockDisplay(Focus.display);
+
+ if (!Focus.terminal_window) {
+ XUnlockDisplay(Focus.display);
+ return;
+ }
+
+ XCloseDisplay(Focus.display);
+ XUnlockDisplay(Focus.display);
+}
+
+#endif /* !__APPLE__ */
diff --git a/src/xtra.h b/src/x11focus.h
similarity index 67%
rename from src/xtra.h
rename to src/x11focus.h
index 7b423af6e..565168d9c 100644
--- a/src/xtra.h
+++ b/src/x11focus.h
@@ -1,7 +1,7 @@
/* xtra.h
*
*
- * Copyright (C) 2014 Toxic All Rights Reserved.
+ * Copyright (C) 2020 Toxic All Rights Reserved.
*
* This file is part of Toxic.
*
@@ -20,21 +20,15 @@
*
*/
-#ifndef XTRA_H
-#define XTRA_H
+#ifndef X11FOCUS_H
+#define X11FOCUS_H
-/* NOTE: If no xlib present don't compile */
-
-typedef enum {
- DT_plain,
- DT_file_list
-}
-DropType;
+#include
-typedef void (*drop_callback)(const char *, DropType);
+/* NOTE: If no xlib present don't compile */
-int init_xtra(drop_callback d);
-void terminate_xtra(void);
-int is_focused(void); /* returns bool */
+int init_x11focus(void);
+void terminate_x11focus(void);
+bool is_focused(void);
-#endif /* XTRA_H */
+#endif /* X11FOCUS */
diff --git a/src/xtra.c b/src/xtra.c
deleted file mode 100644
index 6a3e8907a..000000000
--- a/src/xtra.c
+++ /dev/null
@@ -1,426 +0,0 @@
-/* xtra.c
- *
- *
- * Copyright (C) 2014 Toxic All Rights Reserved.
- *
- * This file is part of Toxic.
- *
- * Toxic is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * Toxic is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with Toxic. If not, see .
- *
- */
-
-#include "misc_tools.h"
-#include "xtra.h"
-
-#ifndef __APPLE__
-
-#include
-#include
-
-#include
-#include
-#include
-#include
-#include
-
-
-const Atom XtraTerminate = 1;
-const Atom XtraNil = 0;
-
-static Atom XdndAware;
-static Atom XdndEnter;
-static Atom XdndLeave;
-static Atom XdndPosition;
-static Atom XdndStatus;
-static Atom XdndDrop;
-static Atom XdndSelection;
-static Atom XdndDATA;
-static Atom XdndTypeList;
-static Atom XdndActionCopy;
-static Atom XdndFinished;
-
-struct Xtra {
- drop_callback on_drop;
- Display *display;
- Window terminal_window;
- Window proxy_window;
- Window source_window; /* When we have a drop */
- Atom handling_version;
- Atom expecting_type;
-} Xtra;
-
-typedef struct Property {
- unsigned char *data;
- int read_format;
- unsigned long read_num;
- Atom read_type;
-} Property;
-
-static Property read_property(Window s, Atom p)
-{
- Atom read_type;
- int read_format;
- unsigned long read_num;
- unsigned long left_bytes;
- unsigned char *data = NULL;
-
- int read_bytes = 1024;
-
- /* Keep trying to read the property until there are no bytes unread */
- do {
- if (data) {
- XFree(data);
- }
-
- XGetWindowProperty(Xtra.display, s,
- p, 0,
- read_bytes,
- False, AnyPropertyType,
- &read_type, &read_format,
- &read_num, &left_bytes,
- &data);
-
- read_bytes *= 2;
- } while (left_bytes != 0);
-
- Property property = {data, read_format, read_num, read_type};
- return property;
-}
-
-static Atom get_dnd_type(long *a, int l)
-{
- int i = 0;
-
- for (; i < l; i ++) {
- if (a[i] != XtraNil) {
- return a[i]; /* Get first valid */
- }
- }
-
- return XtraNil;
-}
-
-/* TODO maybe support only certain types in the future */
-static void handle_xdnd_enter(XClientMessageEvent *e)
-{
- Xtra.handling_version = (e->data.l[1] >> 24);
-
- if ((e->data.l[1] & 1)) {
- // Fetch the list of possible conversions
- Property p = read_property(e->data.l[0], XdndTypeList);
- Xtra.expecting_type = get_dnd_type((long *)p.data, p.read_num);
- XFree(p.data);
- } else {
- // Use the available list
- Xtra.expecting_type = get_dnd_type(e->data.l + 2, 3);
- }
-}
-
-static void handle_xdnd_position(XClientMessageEvent *e)
-{
- XEvent ev = {
- .xclient = {
- .type = ClientMessage,
- .display = e->display,
- .window = e->data.l[0],
- .message_type = XdndStatus,
- .format = 32,
- .data = {
- .l = {
- Xtra.proxy_window,
- (Xtra.expecting_type != XtraNil),
- 0, 0,
- XdndActionCopy
- }
- }
- }
- };
-
- XSendEvent(Xtra.display, e->data.l[0], False, NoEventMask, &ev);
- XFlush(Xtra.display);
-}
-
-static void handle_xdnd_drop(XClientMessageEvent *e)
-{
- /* Not expecting any type */
- if (Xtra.expecting_type == XtraNil) {
- XEvent ev = {
- .xclient = {
- .type = ClientMessage,
- .display = e->display,
- .window = e->data.l[0],
- .message_type = XdndFinished,
- .format = 32,
- .data = {
- .l = {Xtra.proxy_window, 0, 0}
- }
- }
- };
-
- XSendEvent(Xtra.display, e->data.l[0], False, NoEventMask, &ev);
- } else {
- Xtra.source_window = e->data.l[0];
- XConvertSelection(Xtra.display,
- XdndSelection,
- Xtra.expecting_type,
- XdndSelection,
- Xtra.proxy_window,
- Xtra.handling_version >= 1 ? e->data.l[2] : CurrentTime);
- }
-}
-
-static void handle_xdnd_selection(XSelectionEvent *e)
-{
- UNUSED_VAR(e);
-
- /* DnD succesfully finished, send finished and call callback */
- XEvent ev = {
- .xclient = {
- .type = ClientMessage,
- .display = Xtra.display,
- .window = Xtra.source_window,
- .message_type = XdndFinished,
- .format = 32,
- .data = {
- .l = {Xtra.proxy_window, 1, XdndActionCopy}
- }
- }
- };
- XSendEvent(Xtra.display, Xtra.source_window, False, NoEventMask, &ev);
-
- Property p = read_property(Xtra.proxy_window, XdndSelection);
- DropType dt;
-
- if (strcmp(XGetAtomName(Xtra.display, p.read_type), "text/uri-list") == 0) {
- dt = DT_file_list;
- } else { /* text/uri-list */
- dt = DT_plain;
- }
-
-
- /* Call callback for every entry */
- if (Xtra.on_drop && p.read_num) {
- char *sptr;
- char *str = strtok_r((char *) p.data, "\n\r", &sptr);
-
- if (str) {
- Xtra.on_drop(str, dt);
- }
-
- while ((str = strtok_r(NULL, "\n\r", &sptr))) {
- Xtra.on_drop(str, dt);
- }
- }
-
- if (p.data) {
- XFree(p.data);
- }
-}
-
-void *event_loop(void *p)
-{
- /* Handle events like a real nigga */
-
- UNUSED_VAR(p); /* DINDUNOTHIN */
-
- XEvent event;
-
- while (Xtra.display) {
- /* NEEDMOEVENTSFODEMPROGRAMS */
-
- int pending = 0;
-
- XLockDisplay(Xtra.display);
-
- if ((pending = XPending(Xtra.display))) {
- XNextEvent(Xtra.display, &event);
- }
-
- if (!pending) {
- XUnlockDisplay(Xtra.display);
- sleep_thread(10000L);
- continue;
- }
-
- if (event.type == ClientMessage) {
- Atom type = event.xclient.message_type;
-
- if (type == XdndEnter) {
- handle_xdnd_enter(&event.xclient);
- } else if (type == XdndPosition) {
- handle_xdnd_position(&event.xclient);
- } else if (type == XdndDrop) {
- handle_xdnd_drop(&event.xclient);
- } else if (type == XtraTerminate) {
- break;
- }
- } else if (event.type == SelectionNotify) {
- handle_xdnd_selection(&event.xselection);
- }
- /* AINNOBODYCANHANDLEDEMEVENTS*/
- else {
- XSendEvent(Xtra.display, Xtra.terminal_window, 0, 0, &event);
- }
-
- XUnlockDisplay(Xtra.display);
- }
-
- /* Actual XTRA termination
- * Please call xtra_terminate() at exit
- * otherwise HEWUSAGUDBOI happens
- */
- if (Xtra.display) {
- XCloseDisplay(Xtra.display);
- }
-
- return (Xtra.display = NULL);
-}
-
-static long unsigned int focused_window_id(void)
-{
- if (!Xtra.display) {
- return 0;
- }
-
- Window focus;
- int revert;
- XLockDisplay(Xtra.display);
- XGetInputFocus(Xtra.display, &focus, &revert);
- XUnlockDisplay(Xtra.display);
- return focus;
-}
-
-int is_focused(void)
-{
- return Xtra.display && (Xtra.proxy_window == focused_window_id() || Xtra.terminal_window == focused_window_id());
-}
-
-int init_xtra(drop_callback d)
-{
- if (!d) {
- return -1;
- } else {
- Xtra.on_drop = d;
- }
-
- XInitThreads();
-
- if (!(Xtra.display = XOpenDisplay(NULL))) {
- return -1;
- }
-
- Xtra.terminal_window = focused_window_id();
-
- /* OSX: if focused window is 0, it means toxic is ran from
- * native terminal and not X11 terminal window, silently exit */
- if (!Xtra.terminal_window) {
- return 0;
- }
-
- {
- /* Create an invisible window which will act as proxy for the DnD operation. */
- XSetWindowAttributes attr = {0};
- attr.event_mask = EnterWindowMask |
- LeaveWindowMask |
- ButtonMotionMask |
- ButtonPressMask |
- ButtonReleaseMask |
- ResizeRedirectMask;
-
- attr.do_not_propagate_mask = NoEventMask;
-
- Window root;
- int x, y;
- unsigned int wht, hht, b, d;
-
- /* Since we cannot capture resize events for parent window we will have to create
- * this window to have maximum size as defined in root window
- */
- XGetGeometry(Xtra.display,
- XDefaultRootWindow(Xtra.display),
- &root, &x, &y, &wht, &hht, &b, &d);
-
- if (!(Xtra.proxy_window = XCreateWindow
- (Xtra.display, Xtra.terminal_window, /* Parent */
- 0, 0, /* Position */
- wht, hht, /* Width + height */
- 0, /* Border width */
- CopyFromParent, /* Depth */
- InputOnly, /* Class */
- CopyFromParent, /* Visual */
- CWEventMask | CWCursor, /* Value mask */
- &attr))) { /* Attributes for value mask */
- return -1;
- }
- }
-
- XMapWindow(Xtra.display, Xtra.proxy_window); /* Show window (sandwich) */
- XLowerWindow(Xtra.display, Xtra.proxy_window); /* Don't interfere with parent lmao */
-
- XdndAware = XInternAtom(Xtra.display, "XdndAware", False);
- XdndEnter = XInternAtom(Xtra.display, "XdndEnter", False);
- XdndLeave = XInternAtom(Xtra.display, "XdndLeave", False);
- XdndPosition = XInternAtom(Xtra.display, "XdndPosition", False);
- XdndStatus = XInternAtom(Xtra.display, "XdndStatus", False);
- XdndDrop = XInternAtom(Xtra.display, "XdndDrop", False);
- XdndSelection = XInternAtom(Xtra.display, "XdndSelection", False);
- XdndDATA = XInternAtom(Xtra.display, "XdndDATA", False);
- XdndTypeList = XInternAtom(Xtra.display, "XdndTypeList", False);
- XdndActionCopy = XInternAtom(Xtra.display, "XdndActionCopy", False);
- XdndFinished = XInternAtom(Xtra.display, "XdndFinished", False);
-
- /* Inform my nigga windows that we are aware of dnd */
- Atom XdndVersion = 3;
- XChangeProperty(Xtra.display,
- Xtra.proxy_window,
- XdndAware,
- XA_ATOM,
- 32,
- PropModeReplace,
- (unsigned char *)&XdndVersion, 1);
-
- pthread_t id;
-
- if (pthread_create(&id, NULL, event_loop, NULL) != 0) {
- return -1;
- }
-
- pthread_detach(id);
-
- return 0;
-}
-
-void terminate_xtra(void)
-{
- if (!Xtra.display || !Xtra.terminal_window) {
- return;
- }
-
- XEvent terminate = {
- .xclient = {
- .type = ClientMessage,
- .display = Xtra.display,
- .message_type = XtraTerminate,
- }
- };
-
- XLockDisplay(Xtra.display);
- XDeleteProperty(Xtra.display, Xtra.proxy_window, XdndAware);
- XSendEvent(Xtra.display, Xtra.proxy_window, 0, NoEventMask, &terminate);
- XUnlockDisplay(Xtra.display);
-
- while (Xtra.display); /* Wait for termination */
-}
-
-#endif /* !__APPLE__ */