mirror of
https://github.com/Llewellynvdm/conky.git
synced 2025-01-13 19:22:58 +00:00
Fix X11 area enter & leave bug
Signed-off-by: Tin <tin.svagelj@live.com>
This commit is contained in:
parent
bf900545c0
commit
a4ac632db7
@ -169,7 +169,6 @@ if(BUILD_X11)
|
|||||||
option(BUILD_XINERAMA "Build Xinerama support" true)
|
option(BUILD_XINERAMA "Build Xinerama support" true)
|
||||||
option(BUILD_XDBE "Build Xdbe (double-buffer) support" true)
|
option(BUILD_XDBE "Build Xdbe (double-buffer) support" true)
|
||||||
option(BUILD_XFT "Build Xft (freetype fonts) support" true)
|
option(BUILD_XFT "Build Xft (freetype fonts) support" true)
|
||||||
option(BUILD_IMLIB2 "Enable Imlib2 support" true)
|
|
||||||
option(BUILD_XSHAPE "Enable Xshape support" true)
|
option(BUILD_XSHAPE "Enable Xshape support" true)
|
||||||
else(BUILD_X11)
|
else(BUILD_X11)
|
||||||
set(OWN_WINDOW false CACHE BOOL "Enable own_window support" FORCE)
|
set(OWN_WINDOW false CACHE BOOL "Enable own_window support" FORCE)
|
||||||
@ -178,7 +177,6 @@ else(BUILD_X11)
|
|||||||
set(BUILD_XINERAMA false CACHE BOOL "Build Xinerama support" FORCE)
|
set(BUILD_XINERAMA false CACHE BOOL "Build Xinerama support" FORCE)
|
||||||
set(BUILD_XDBE false CACHE BOOL "Build Xdbe (double-buffer) support" FORCE)
|
set(BUILD_XDBE false CACHE BOOL "Build Xdbe (double-buffer) support" FORCE)
|
||||||
set(BUILD_XFT false CACHE BOOL "Build Xft (freetype fonts) support" FORCE)
|
set(BUILD_XFT false CACHE BOOL "Build Xft (freetype fonts) support" FORCE)
|
||||||
set(BUILD_IMLIB2 false CACHE BOOL "Enable Imlib2 support" FORCE)
|
|
||||||
set(BUILD_XSHAPE false CACHE BOOL "Enable Xshape support" FORCE)
|
set(BUILD_XSHAPE false CACHE BOOL "Enable Xshape support" FORCE)
|
||||||
set(BUILD_NVIDIA false)
|
set(BUILD_NVIDIA false)
|
||||||
endif(BUILD_X11)
|
endif(BUILD_X11)
|
||||||
@ -192,9 +190,14 @@ if(BUILD_WAYLAND)
|
|||||||
endif(BUILD_WAYLAND)
|
endif(BUILD_WAYLAND)
|
||||||
|
|
||||||
if(BUILD_GUI)
|
if(BUILD_GUI)
|
||||||
|
option(BUILD_IMLIB2 "Enable Imlib2 support" true)
|
||||||
option(BUILD_MOUSE_EVENTS "Enable mouse event support" true)
|
option(BUILD_MOUSE_EVENTS "Enable mouse event support" true)
|
||||||
endif(BUILD_GUI)
|
endif(BUILD_GUI)
|
||||||
|
|
||||||
|
if(BUILD_GUI AND BUILD_MOUSE_EVENTS)
|
||||||
|
option(BUILD_XINPUT "Build Xinput 2 support" true)
|
||||||
|
endif(BUILD_GUI AND BUILD_MOUSE_EVENTS)
|
||||||
|
|
||||||
if(OWN_WINDOW)
|
if(OWN_WINDOW)
|
||||||
option(BUILD_ARGB "Build ARGB (real transparency) support" true)
|
option(BUILD_ARGB "Build ARGB (real transparency) support" true)
|
||||||
else(OWN_WINDOW)
|
else(OWN_WINDOW)
|
||||||
|
@ -402,6 +402,29 @@ if(BUILD_X11)
|
|||||||
|
|
||||||
set(conky_libs ${conky_libs} ${X11_Xfixes_LIB})
|
set(conky_libs ${conky_libs} ${X11_Xfixes_LIB})
|
||||||
endif(BUILD_XFIXES)
|
endif(BUILD_XFIXES)
|
||||||
|
|
||||||
|
# check for Xinput
|
||||||
|
if(BUILD_XINPUT)
|
||||||
|
if(NOT X11_Xinput_FOUND)
|
||||||
|
message(FATAL_ERROR "Unable to find Xinput library")
|
||||||
|
endif(NOT X11_Xinput_FOUND)
|
||||||
|
|
||||||
|
set(conky_libs ${conky_libs} ${X11_Xinput_LIB})
|
||||||
|
endif(BUILD_XINPUT)
|
||||||
|
|
||||||
|
|
||||||
|
if(X11_xcb_FOUND)
|
||||||
|
set(HAVE_XCB true)
|
||||||
|
set(conky_libs ${conky_libs} ${X11_xcb_LIB})
|
||||||
|
if(X11_xcb_errors_FOUND)
|
||||||
|
set(HAVE_XCB_ERRORS true)
|
||||||
|
set(conky_libs ${conky_libs} ${X11_xcb_LIB} ${X11_xcb_errors_LIB})
|
||||||
|
else(X11_xcb_errors_FOUND)
|
||||||
|
set(HAVE_XCB_ERRORS false)
|
||||||
|
endif(X11_xcb_errors_FOUND)
|
||||||
|
else(X11_xcb_FOUND)
|
||||||
|
set(HAVE_XCB false)
|
||||||
|
endif(X11_xcb_FOUND)
|
||||||
else(X11_FOUND)
|
else(X11_FOUND)
|
||||||
message(FATAL_ERROR "Unable to find X11 library")
|
message(FATAL_ERROR "Unable to find X11 library")
|
||||||
endif(X11_FOUND)
|
endif(X11_FOUND)
|
||||||
|
@ -42,6 +42,9 @@
|
|||||||
|
|
||||||
#cmakedefine HAVE_CLOCK_GETTIME 1
|
#cmakedefine HAVE_CLOCK_GETTIME 1
|
||||||
|
|
||||||
|
#cmakedefine HAVE_XCB 1
|
||||||
|
#cmakedefine HAVE_XCB_ERRORS 1
|
||||||
|
|
||||||
#cmakedefine BUILD_WAYLAND 1
|
#cmakedefine BUILD_WAYLAND 1
|
||||||
|
|
||||||
#cmakedefine BUILD_X11 1
|
#cmakedefine BUILD_X11 1
|
||||||
@ -60,6 +63,8 @@
|
|||||||
|
|
||||||
#cmakedefine BUILD_XFIXES 1
|
#cmakedefine BUILD_XFIXES 1
|
||||||
|
|
||||||
|
#cmakedefine BUILD_XINPUT 1
|
||||||
|
|
||||||
#cmakedefine BUILD_ARGB 1
|
#cmakedefine BUILD_ARGB 1
|
||||||
|
|
||||||
#cmakedefine BUILD_XDBE 1
|
#cmakedefine BUILD_XDBE 1
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <X11/extensions/XI2.h>
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
#ifdef BUILD_X11
|
#ifdef BUILD_X11
|
||||||
@ -44,6 +45,9 @@
|
|||||||
#endif /* BUILD_IMLIB2 */
|
#endif /* BUILD_IMLIB2 */
|
||||||
#ifdef BUILD_MOUSE_EVENTS
|
#ifdef BUILD_MOUSE_EVENTS
|
||||||
#include "mouse-events.h"
|
#include "mouse-events.h"
|
||||||
|
#ifdef BUILD_XINPUT
|
||||||
|
#include <X11/extensions/XInput2.h>
|
||||||
|
#endif /* BUILD_XINPUT */
|
||||||
#endif /* BUILD_MOUSE_EVENTS */
|
#endif /* BUILD_MOUSE_EVENTS */
|
||||||
#endif /* BUILD_X11 */
|
#endif /* BUILD_X11 */
|
||||||
|
|
||||||
@ -51,6 +55,7 @@
|
|||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include "colours.h"
|
#include "colours.h"
|
||||||
#include "conky.h"
|
#include "conky.h"
|
||||||
@ -229,27 +234,6 @@ bool display_output_x11::shutdown() {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void propagate_unconsumed_event(XEvent &ev, bool is_consumed) {
|
|
||||||
const uint32_t capture_mask = ButtonPressMask | ButtonReleaseMask | ButtonMotionMask;
|
|
||||||
|
|
||||||
// SAFETY: XEvent is a union and all event types this function gets called for
|
|
||||||
// share same alignment and accessed fields share same offsets.
|
|
||||||
if (!is_consumed) {
|
|
||||||
/* forward the event to the desktop window */
|
|
||||||
ev.xmotion.window = window.desktop;
|
|
||||||
ev.xmotion.x = ev.xmotion.x_root;
|
|
||||||
ev.xmotion.y = ev.xmotion.y_root;
|
|
||||||
XSendEvent(display, window.desktop, False, capture_mask, &ev);
|
|
||||||
if (false && ev.type == ButtonPress) {
|
|
||||||
XSetInputFocus(display, window.desktop, RevertToNone, ev.xmotion.time);
|
|
||||||
} else if (false && ev.type == MotionNotify) {
|
|
||||||
XSetInputFocus(display, window.window, RevertToParent, ev.xmotion.time);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
XSetInputFocus(display, window.window, RevertToParent, ev.xmotion.time);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool display_output_x11::main_loop_wait(double t) {
|
bool display_output_x11::main_loop_wait(double t) {
|
||||||
/* wait for X event or timeout */
|
/* wait for X event or timeout */
|
||||||
if (!display || !window.gc) return true;
|
if (!display || !window.gc) return true;
|
||||||
@ -397,6 +381,50 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
bool consumed = false;
|
bool consumed = false;
|
||||||
|
|
||||||
XNextEvent(display, &ev);
|
XNextEvent(display, &ev);
|
||||||
|
|
||||||
|
#ifdef BUILD_XINPUT
|
||||||
|
if (ev.type == GenericEvent && ev.xcookie.extension == window.xi_opcode) {
|
||||||
|
XGetEventData(display, &ev.xcookie);
|
||||||
|
if (ev.xcookie.evtype == XI_Motion) {
|
||||||
|
auto *data = reinterpret_cast<XIDeviceEvent*>(ev.xcookie.data);
|
||||||
|
|
||||||
|
// XQueryPointer returns wrong results because conky is a weird window
|
||||||
|
Window query_result = query_x11_window_at_pos(display, data->root_x, data->root_y);
|
||||||
|
|
||||||
|
//printf("over: %#x w:%#x d:%#x r:%#x\n", query_result, window.window, window.desktop, window.root);
|
||||||
|
|
||||||
|
static bool cursor_inside = false;
|
||||||
|
if ((query_result != 0 && query_result == window.window) ||
|
||||||
|
((query_result == window.desktop || query_result == window.root || query_result == 0) && data->root_x >= window.x && data->root_x < (window.x + window.width) &&
|
||||||
|
data->root_y >= window.y && data->root_y < (window.y + window.height))) {
|
||||||
|
//puts("over self");
|
||||||
|
if (!cursor_inside) {
|
||||||
|
llua_mouse_hook(mouse_crossing_event(
|
||||||
|
mouse_event_t::AREA_ENTER,
|
||||||
|
data->root_x - window.x, data->root_y - window.x,
|
||||||
|
data->root_x, data->root_y
|
||||||
|
));
|
||||||
|
}
|
||||||
|
cursor_inside = true;
|
||||||
|
} else if (cursor_inside) {
|
||||||
|
//puts("left");
|
||||||
|
llua_mouse_hook(mouse_crossing_event(
|
||||||
|
mouse_event_t::AREA_LEAVE,
|
||||||
|
data->root_x - window.x, data->root_y - window.x,
|
||||||
|
data->root_x, data->root_y
|
||||||
|
));
|
||||||
|
cursor_inside = false;
|
||||||
|
} else {
|
||||||
|
//printf("not over self\n");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
XFreeEventData(display, &ev.xcookie);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
#endif /* BUILD_XINPUT */
|
||||||
|
|
||||||
|
// Any of the remaining events apply to conky window
|
||||||
|
if (ev.xany.window != window.window) continue;
|
||||||
switch (ev.type) {
|
switch (ev.type) {
|
||||||
case Expose: {
|
case Expose: {
|
||||||
XRectangle r;
|
XRectangle r;
|
||||||
@ -406,6 +434,10 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
r.height = ev.xexpose.height;
|
r.height = ev.xexpose.height;
|
||||||
XUnionRectWithRegion(&r, x11_stuff.region, x11_stuff.region);
|
XUnionRectWithRegion(&r, x11_stuff.region, x11_stuff.region);
|
||||||
XSync(display, False);
|
XSync(display, False);
|
||||||
|
|
||||||
|
// modify for propagation
|
||||||
|
ev.xexpose.x += window.x;
|
||||||
|
ev.xexpose.y += window.y;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -427,7 +459,7 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
#ifdef USE_ARGB
|
#ifdef USE_ARGB
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
break;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef OWN_WINDOW
|
#ifdef OWN_WINDOW
|
||||||
@ -436,7 +468,7 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
if (own_window.get(*state)) {
|
if (own_window.get(*state)) {
|
||||||
set_transparent_background(window.window);
|
set_transparent_background(window.window);
|
||||||
}
|
}
|
||||||
break;
|
continue;
|
||||||
|
|
||||||
case ConfigureNotify:
|
case ConfigureNotify:
|
||||||
if (own_window.get(*state)) {
|
if (own_window.get(*state)) {
|
||||||
@ -477,7 +509,7 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
fixed_pos = 1;
|
fixed_pos = 1;
|
||||||
} */
|
} */
|
||||||
}
|
}
|
||||||
break;
|
continue;
|
||||||
|
|
||||||
case ButtonPress:
|
case ButtonPress:
|
||||||
#ifdef BUILD_MOUSE_EVENTS
|
#ifdef BUILD_MOUSE_EVENTS
|
||||||
@ -519,7 +551,6 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
/* allow conky to hold input focus. */
|
/* allow conky to hold input focus. */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
propagate_unconsumed_event(ev, consumed);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -553,7 +584,6 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
/* allow conky to hold input focus. */
|
/* allow conky to hold input focus. */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
propagate_unconsumed_event(ev, consumed);
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
#ifdef BUILD_MOUSE_EVENTS
|
#ifdef BUILD_MOUSE_EVENTS
|
||||||
@ -565,30 +595,25 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
consumed = llua_mouse_hook(mouse_move_event(
|
consumed = llua_mouse_hook(mouse_move_event(
|
||||||
ev.xmotion.x, ev.xmotion.y, ev.xmotion.x_root, ev.xmotion.y_root, ev.xmotion.state
|
ev.xmotion.x, ev.xmotion.y, ev.xmotion.x_root, ev.xmotion.y_root, ev.xmotion.state
|
||||||
));
|
));
|
||||||
propagate_unconsumed_event(ev, consumed);
|
|
||||||
break;
|
break;
|
||||||
case LeaveNotify:
|
case LeaveNotify:
|
||||||
XUngrabPointer(display, ev.xcrossing.time);
|
|
||||||
case EnterNotify:
|
case EnterNotify:
|
||||||
{
|
if (window.xi_opcode == 0) {
|
||||||
bool not_over_conky = ev.xcrossing.x_root <= window.x || ev.xcrossing.y_root <= window.y ||
|
bool not_over_conky = ev.xcrossing.x_root <= window.x || ev.xcrossing.y_root <= window.y ||
|
||||||
ev.xcrossing.x_root >= window.x + window.width || ev.xcrossing.y_root >= window.y + window.height;
|
ev.xcrossing.x_root >= window.x + window.width || ev.xcrossing.y_root >= window.y + window.height;
|
||||||
static bool pointer_inside = false;
|
|
||||||
|
|
||||||
if ((not_over_conky && ev.xcrossing.type == LeaveNotify) ||
|
if ((not_over_conky && ev.xcrossing.type == LeaveNotify) ||
|
||||||
(!pointer_inside && ev.xcrossing.type == EnterNotify)) {
|
(!not_over_conky && ev.xcrossing.type == EnterNotify)) {
|
||||||
llua_mouse_hook(mouse_crossing_event(
|
llua_mouse_hook(mouse_crossing_event(
|
||||||
ev.xcrossing.type == EnterNotify ? mouse_event_t::AREA_ENTER : mouse_event_t::AREA_LEAVE,
|
ev.xcrossing.type == EnterNotify ? mouse_event_t::AREA_ENTER : mouse_event_t::AREA_LEAVE,
|
||||||
ev.xcrossing.x, ev.xcrossing.y, ev.xcrossing.x_root, ev.xcrossing.y_root
|
ev.xcrossing.x, ev.xcrossing.y, ev.xcrossing.x_root, ev.xcrossing.y_root
|
||||||
));
|
));
|
||||||
pointer_inside = ev.xcrossing.type == EnterNotify;
|
|
||||||
// can't propagate these events in a way that makes sense
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
// can't propagate these events in a way that makes sense for desktop
|
||||||
|
continue;
|
||||||
#endif /* BUILD_MOUSE_EVENTS */
|
#endif /* BUILD_MOUSE_EVENTS */
|
||||||
#endif
|
#endif /* OWN_WINDOW */
|
||||||
|
|
||||||
default:
|
default:
|
||||||
#ifdef BUILD_XDAMAGE
|
#ifdef BUILD_XDAMAGE
|
||||||
if (ev.type == x11_stuff.event_base + XDamageNotify) {
|
if (ev.type == x11_stuff.event_base + XDamageNotify) {
|
||||||
@ -597,10 +622,20 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
XFixesSetRegion(display, x11_stuff.part, &dev->area, 1);
|
XFixesSetRegion(display, x11_stuff.part, &dev->area, 1);
|
||||||
XFixesUnionRegion(display, x11_stuff.region2, x11_stuff.region2,
|
XFixesUnionRegion(display, x11_stuff.region2, x11_stuff.region2,
|
||||||
x11_stuff.part);
|
x11_stuff.part);
|
||||||
|
continue; // TODO: Propagate damage
|
||||||
}
|
}
|
||||||
#endif /* BUILD_XDAMAGE */
|
#endif /* BUILD_XDAMAGE */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!consumed) {
|
||||||
|
propagate_x11_event(ev);
|
||||||
|
} else {
|
||||||
|
InputEvent *i_ev = xev_as_input_event(ev);
|
||||||
|
if (i_ev != nullptr) {
|
||||||
|
XSetInputFocus(display, window.window, RevertToParent, i_ev->common.time);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef BUILD_XDAMAGE
|
#ifdef BUILD_XDAMAGE
|
||||||
@ -648,6 +683,7 @@ bool display_output_x11::main_loop_wait(double t) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void display_output_x11::sigterm_cleanup() {
|
void display_output_x11::sigterm_cleanup() {
|
||||||
|
puts("sigtermed");
|
||||||
XDestroyRegion(x11_stuff.region);
|
XDestroyRegion(x11_stuff.region);
|
||||||
x11_stuff.region = nullptr;
|
x11_stuff.region = nullptr;
|
||||||
#ifdef BUILD_XDAMAGE
|
#ifdef BUILD_XDAMAGE
|
||||||
@ -660,6 +696,7 @@ void display_output_x11::sigterm_cleanup() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void display_output_x11::cleanup() {
|
void display_output_x11::cleanup() {
|
||||||
|
puts("normal cleanup");
|
||||||
if (window_created == 1) {
|
if (window_created == 1) {
|
||||||
int border_total = get_border_total();
|
int border_total = get_border_total();
|
||||||
|
|
||||||
|
@ -31,6 +31,7 @@
|
|||||||
#define _LOGGING_H
|
#define _LOGGING_H
|
||||||
|
|
||||||
#include <cstdio>
|
#include <cstdio>
|
||||||
|
#include <cinttypes>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "i18n.h"
|
#include "i18n.h"
|
||||||
@ -81,6 +82,7 @@ void NORM_ERR(const char *format, Args &&...args) {
|
|||||||
|
|
||||||
/* critical error with additional cleanup */
|
/* critical error with additional cleanup */
|
||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
|
__attribute__((noreturn))
|
||||||
inline void CRIT_ERR_FREE(void *memtofree1, void *memtofree2,
|
inline void CRIT_ERR_FREE(void *memtofree1, void *memtofree2,
|
||||||
const char *format, Args &&...args) {
|
const char *format, Args &&...args) {
|
||||||
NORM_ERR(format, args...);
|
NORM_ERR(format, args...);
|
||||||
@ -92,6 +94,7 @@ inline void CRIT_ERR_FREE(void *memtofree1, void *memtofree2,
|
|||||||
|
|
||||||
/* critical error */
|
/* critical error */
|
||||||
template <typename... Args>
|
template <typename... Args>
|
||||||
|
__attribute__((noreturn))
|
||||||
inline void CRIT_ERR(const char *format, Args &&...args) {
|
inline void CRIT_ERR(const char *format, Args &&...args) {
|
||||||
CRIT_ERR_FREE(nullptr, nullptr, format, args...);
|
CRIT_ERR_FREE(nullptr, nullptr, format, args...);
|
||||||
}
|
}
|
||||||
|
217
src/x11.cc
217
src/x11.cc
@ -28,10 +28,18 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <X11/X.h>
|
#include <X11/X.h>
|
||||||
|
#include <sys/types.h>
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "conky.h"
|
#include "conky.h"
|
||||||
#include "logging.h"
|
#include "logging.h"
|
||||||
|
#include "x11.h"
|
||||||
|
|
||||||
|
#include <cstddef>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstring>
|
||||||
|
#include <string>
|
||||||
|
#include <cstdlib>
|
||||||
|
|
||||||
#pragma GCC diagnostic push
|
#pragma GCC diagnostic push
|
||||||
#pragma GCC diagnostic ignored "-Wvariadic-macros"
|
#pragma GCC diagnostic ignored "-Wvariadic-macros"
|
||||||
@ -61,10 +69,24 @@
|
|||||||
#ifdef BUILD_XFIXES
|
#ifdef BUILD_XFIXES
|
||||||
#include <X11/extensions/Xfixes.h>
|
#include <X11/extensions/Xfixes.h>
|
||||||
#endif /* BUILD_XFIXES */
|
#endif /* BUILD_XFIXES */
|
||||||
|
#ifdef BUILD_XINPUT
|
||||||
|
#include <vector>
|
||||||
|
#include <X11/extensions/XInput2.h>
|
||||||
|
#endif /* BUILD_XINPUT */
|
||||||
|
#ifdef HAVE_XCB_ERRORS
|
||||||
|
#include <xcb/xcb.h>
|
||||||
|
#include <xcb/xcb_errors.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
/* some basic X11 stuff */
|
/* some basic X11 stuff */
|
||||||
Display *display = nullptr;
|
Display *display = nullptr;
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef HAVE_XCB_ERRORS
|
||||||
|
xcb_connection_t *xcb_connection;
|
||||||
|
xcb_errors_context_t *xcb_errors_ctx;
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Window stuff */
|
/* Window stuff */
|
||||||
struct conky_x11_window window;
|
struct conky_x11_window window;
|
||||||
|
|
||||||
@ -215,20 +237,77 @@ imlib_cache_size_setting imlib_cache_size;
|
|||||||
/******************** </SETTINGS> ************************/
|
/******************** </SETTINGS> ************************/
|
||||||
|
|
||||||
/* WARNING, this type not in Xlib spec */
|
/* WARNING, this type not in Xlib spec */
|
||||||
static int __attribute__((noreturn))
|
static int
|
||||||
x11_error_handler(Display *d, XErrorEvent *err) {
|
x11_error_handler(Display *d, XErrorEvent *err) {
|
||||||
NORM_ERR(
|
char *error_name = nullptr;
|
||||||
"X Error: type %i Display %lx XID %li serial %lu error_code %i "
|
bool name_allocated = false;
|
||||||
"request_code %i minor_code %i other Display: %lx\n",
|
|
||||||
err->type, (long unsigned)err->display,
|
char *code_description = nullptr;
|
||||||
static_cast<long>(err->resourceid), err->serial, err->error_code,
|
bool code_allocated = false;
|
||||||
err->request_code, err->minor_code, (long unsigned)d);
|
|
||||||
abort();
|
#ifdef HAVE_XCB_ERRORS
|
||||||
|
if (xcb_errors_ctx != nullptr) {
|
||||||
|
const char *extension;
|
||||||
|
const char *base_name = xcb_errors_get_name_for_error(xcb_errors_ctx, err->error_code, &extension);
|
||||||
|
if (extension != nullptr) {
|
||||||
|
error_name = new char(strlen(base_name) + strlen(extension) + 4);
|
||||||
|
sprintf(error_name, "%s (%s)", base_name, extension);
|
||||||
|
name_allocated = true;
|
||||||
|
} else {
|
||||||
|
error_name = const_cast<char*>(base_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int __attribute__((noreturn)) x11_ioerror_handler(Display *d) {
|
const char *major = xcb_errors_get_name_for_major_code(xcb_errors_ctx, err->request_code);
|
||||||
NORM_ERR("X Error: Display %lx\n", (long unsigned)d);
|
const char *minor = xcb_errors_get_name_for_minor_code(xcb_errors_ctx, err->request_code, err->minor_code);
|
||||||
exit(1);
|
if (minor != nullptr) {
|
||||||
|
code_description = new char(strlen(base_name) + strlen(extension) + 4);
|
||||||
|
sprintf(code_description, "%s - %s", major, minor);
|
||||||
|
code_allocated = true;
|
||||||
|
} else {
|
||||||
|
code_description = const_cast<char*>(major);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (error_name == nullptr) {
|
||||||
|
if (err->error_code > 0 && err->error_code < 17) {
|
||||||
|
static std::array<std::string, 17> NAMES = {
|
||||||
|
"request", "value", "window", "pixmap", "atom", "cursor", "font", "match",
|
||||||
|
"drawable", "access", "alloc", "colormap", "G context", "ID choice",
|
||||||
|
"name", "length", "implementation"
|
||||||
|
};
|
||||||
|
error_name = const_cast<char*>(NAMES[err->error_code].c_str());
|
||||||
|
} else {
|
||||||
|
static char code_name_buffer[3];
|
||||||
|
error_name = reinterpret_cast<char*>(&code_name_buffer);
|
||||||
|
sprintf(error_name, "%d", err->error_code);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (code_description == nullptr) {
|
||||||
|
code_description = new char(30 + 6 + 1);
|
||||||
|
sprintf(code_description, "error code: [major: %i, minor: %i]", err->request_code, err->minor_code);
|
||||||
|
code_allocated = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
NORM_ERR(
|
||||||
|
"X %s Error:\n"
|
||||||
|
"Display: %lx, XID: %li, Serial: %lu\n"
|
||||||
|
"%s",
|
||||||
|
error_name,
|
||||||
|
reinterpret_cast<uint64_t>(err->display),
|
||||||
|
static_cast<int64_t>(err->resourceid), err->serial,
|
||||||
|
code_description
|
||||||
|
);
|
||||||
|
|
||||||
|
if (name_allocated) free(error_name);
|
||||||
|
if (code_allocated) free(code_description);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
__attribute__((noreturn))
|
||||||
|
static int x11_ioerror_handler(Display *d) {
|
||||||
|
CRIT_ERR("X IO Error: Display %lx\n", reinterpret_cast<uint64_t>(d));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* X11 initializer */
|
/* X11 initializer */
|
||||||
@ -267,6 +346,15 @@ static void init_x11() {
|
|||||||
|
|
||||||
update_workarea();
|
update_workarea();
|
||||||
|
|
||||||
|
#ifdef HAVE_XCB_ERRORS
|
||||||
|
auto connection = xcb_connect(NULL, NULL);
|
||||||
|
if (!xcb_connection_has_error(connection)) {
|
||||||
|
if (xcb_errors_context_new(connection, &xcb_errors_ctx) != Success) {
|
||||||
|
xcb_errors_ctx = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif /* HAVE_XCB_ERRORS */
|
||||||
|
|
||||||
/* WARNING, this type not in Xlib spec */
|
/* WARNING, this type not in Xlib spec */
|
||||||
XSetErrorHandler(&x11_error_handler);
|
XSetErrorHandler(&x11_error_handler);
|
||||||
XSetIOErrorHandler(&x11_ioerror_handler);
|
XSetIOErrorHandler(&x11_ioerror_handler);
|
||||||
@ -854,7 +942,7 @@ void x11_init_window(lua::state &l __attribute__((unused)), bool own) {
|
|||||||
|
|
||||||
XFlush(display);
|
XFlush(display);
|
||||||
|
|
||||||
long input_mask = ExposureMask | PropertyChangeMask;
|
int64_t input_mask = ExposureMask | PropertyChangeMask;
|
||||||
#ifdef OWN_WINDOW
|
#ifdef OWN_WINDOW
|
||||||
if (own_window.get(l)) {
|
if (own_window.get(l)) {
|
||||||
input_mask |= StructureNotifyMask | ButtonPressMask | ButtonReleaseMask;
|
input_mask |= StructureNotifyMask | ButtonPressMask | ButtonReleaseMask;
|
||||||
@ -864,10 +952,40 @@ void x11_init_window(lua::state &l __attribute__((unused)), bool own) {
|
|||||||
/* it's not recommended to add event masks to special windows in X; causes a
|
/* it's not recommended to add event masks to special windows in X; causes a
|
||||||
* crash */
|
* crash */
|
||||||
if (own_window_type.get(l) != TYPE_DESKTOP) {
|
if (own_window_type.get(l) != TYPE_DESKTOP) {
|
||||||
input_mask |= ButtonPressMask | ButtonReleaseMask | PointerMotionMask |
|
input_mask |= ButtonPressMask | ButtonReleaseMask | PointerMotionMask;
|
||||||
EnterWindowMask | LeaveWindowMask;
|
}
|
||||||
|
bool xinput_ok = false;
|
||||||
|
#ifdef BUILD_XINPUT
|
||||||
|
do { // not loop
|
||||||
|
int _ignored; // segfault if NULL
|
||||||
|
if (!XQueryExtension(display, "XInputExtension", &window.xi_opcode, &_ignored, &_ignored)) {
|
||||||
|
// events will still ~work but let the user know why they're buggy
|
||||||
|
NORM_ERR("XInput extension is not supported by X11!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t major = 2, minor = 0;
|
||||||
|
uint32_t retval = XIQueryVersion(display, &major, &minor);
|
||||||
|
if (retval != Success) {
|
||||||
|
NORM_ERR("Error: XInput 2.0 is not supported!");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
unsigned char mask_bytes[(XI_LASTEVENT + 7) / 8] = {0}; /* must be zeroed! */
|
||||||
|
XISetMask(mask_bytes, XI_Motion);
|
||||||
|
|
||||||
|
XIEventMask ev_masks[1];
|
||||||
|
ev_masks[0].deviceid = XIAllDevices;
|
||||||
|
ev_masks[0].mask_len = sizeof(mask_bytes);
|
||||||
|
ev_masks[0].mask = mask_bytes;
|
||||||
|
XISelectEvents(display, window.root, ev_masks, 1);
|
||||||
|
xinput_ok = true;
|
||||||
|
} while (false);
|
||||||
|
#endif /* BUILD_XINPUT */
|
||||||
|
if (!xinput_ok && own_window_type.get(l) != TYPE_DESKTOP) {
|
||||||
|
input_mask |= EnterWindowMask | LeaveWindowMask;
|
||||||
}
|
}
|
||||||
#endif /* BUILD_MOUSE_EVENTS */
|
#endif /* BUILD_MOUSE_EVENTS */
|
||||||
|
window.event_mask = input_mask;
|
||||||
XSelectInput(display, window.window, input_mask);
|
XSelectInput(display, window.window, input_mask);
|
||||||
|
|
||||||
window_created = 1;
|
window_created = 1;
|
||||||
@ -1229,3 +1347,74 @@ void print_mouse_speed(struct text_object *obj, char *p,
|
|||||||
XGetPointerControl(display, &acc_num, &acc_denom, &threshold);
|
XGetPointerControl(display, &acc_num, &acc_denom, &threshold);
|
||||||
snprintf(p, p_max_size, "%d%%", (110 - threshold));
|
snprintf(p, p_max_size, "%d%%", (110 - threshold));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
InputEvent *xev_as_input_event(XEvent &ev) {
|
||||||
|
if (ev.type == KeyPress || ev.type == KeyRelease ||
|
||||||
|
ev.type == ButtonPress || ev.type == ButtonRelease || ev.type == MotionNotify ||
|
||||||
|
ev.type == EnterNotify || ev.type == LeaveNotify) {
|
||||||
|
return reinterpret_cast<InputEvent*>(&ev);
|
||||||
|
} else {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void propagate_x11_event(XEvent &ev) {
|
||||||
|
InputEvent *i_ev = xev_as_input_event(ev);
|
||||||
|
/* forward the event to the desktop window */
|
||||||
|
if (i_ev != nullptr) {
|
||||||
|
i_ev->common.window = window.desktop;
|
||||||
|
i_ev->common.x = i_ev->common.x_root;
|
||||||
|
i_ev->common.y = i_ev->common.y_root;
|
||||||
|
}
|
||||||
|
XSendEvent(display, window.desktop, False, window.event_mask, &ev);
|
||||||
|
// FIXME (before-merge): Should we be setting input focus after forwarding
|
||||||
|
// events?
|
||||||
|
if (false && ev.type == ButtonPress) {
|
||||||
|
XSetInputFocus(display, window.desktop, RevertToNone, i_ev->common.time);
|
||||||
|
} else if (false && ev.type == MotionNotify) {
|
||||||
|
XSetInputFocus(display, window.window, RevertToParent, i_ev->common.time);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef BUILD_MOUSE_EVENTS
|
||||||
|
Window last_descendant(Display* display, Window parent) {
|
||||||
|
Window ignored, *children;
|
||||||
|
uint32_t count;
|
||||||
|
|
||||||
|
Window current = parent;
|
||||||
|
|
||||||
|
while (XQueryTree(display, current, &ignored, &ignored, &children, &count) && count != 0) {
|
||||||
|
current = children[count - 1];
|
||||||
|
XFree(children);
|
||||||
|
}
|
||||||
|
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
|
||||||
|
Window query_x11_window_at_pos(Display* display, int x, int y) {
|
||||||
|
Window root = DefaultRootWindow(display);
|
||||||
|
|
||||||
|
Window root_return, parent_return, *windows;
|
||||||
|
unsigned int count;
|
||||||
|
|
||||||
|
Window last = None;
|
||||||
|
|
||||||
|
XWindowAttributes attrs;
|
||||||
|
if (XQueryTree(display, root, &root_return, &parent_return, &windows, &count) != 0) {
|
||||||
|
for (unsigned int i = 0; i < count; i++) {
|
||||||
|
if (XGetWindowAttributes(display, windows[i], &attrs)) {
|
||||||
|
if (attrs.map_state == IsViewable && x >= attrs.x && x < (attrs.x + attrs.width) && y >= attrs.y && y < (attrs.y + attrs.height)) {
|
||||||
|
last = windows[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count != 0) {
|
||||||
|
XFree(windows);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return last_descendant(display, last);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* BUILD_MOUSE_EVENTS */
|
||||||
|
48
src/x11.h
48
src/x11.h
@ -38,6 +38,7 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "setting.hh"
|
#include "setting.hh"
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
#ifdef BUILD_ARGB
|
#ifdef BUILD_ARGB
|
||||||
/* true if use_argb_visual=true and argb visual was found*/
|
/* true if use_argb_visual=true and argb visual was found*/
|
||||||
@ -78,6 +79,9 @@ struct conky_x11_window {
|
|||||||
Colormap colourmap;
|
Colormap colourmap;
|
||||||
GC gc;
|
GC gc;
|
||||||
|
|
||||||
|
// Mask containing all events captured by conky
|
||||||
|
int64_t event_mask;
|
||||||
|
|
||||||
#ifdef BUILD_XDBE
|
#ifdef BUILD_XDBE
|
||||||
XdbeBackBuffer back_buffer;
|
XdbeBackBuffer back_buffer;
|
||||||
#else /*BUILD_XDBE*/
|
#else /*BUILD_XDBE*/
|
||||||
@ -86,6 +90,9 @@ struct conky_x11_window {
|
|||||||
#ifdef BUILD_XFT
|
#ifdef BUILD_XFT
|
||||||
XftDraw *xftdraw;
|
XftDraw *xftdraw;
|
||||||
#endif /*BUILD_XFT*/
|
#endif /*BUILD_XFT*/
|
||||||
|
#ifdef BUILD_XINPUT
|
||||||
|
int32_t xi_opcode;
|
||||||
|
#endif /* BUILD_XINPUT */
|
||||||
|
|
||||||
int width;
|
int width;
|
||||||
int height;
|
int height;
|
||||||
@ -106,6 +113,47 @@ void set_struts(int);
|
|||||||
void x11_init_window(lua::state &l, bool own);
|
void x11_init_window(lua::state &l, bool own);
|
||||||
void deinit_x11();
|
void deinit_x11();
|
||||||
|
|
||||||
|
// Fields common to all X11 input events
|
||||||
|
struct InputEventCommon {
|
||||||
|
int type; /* event type */
|
||||||
|
uint64_t serial; /* # of last request processed by server */
|
||||||
|
Bool send_event; /* true if this came from a SendEvent request */
|
||||||
|
Display *display; /* Display the event was read from */
|
||||||
|
Window window; /* "event" window reported relative to */
|
||||||
|
Window root; /* root window that the event occurred on */
|
||||||
|
Window subwindow; /* child window */
|
||||||
|
Time time; /* milliseconds */
|
||||||
|
int32_t x, y; /* pointer x, y coordinates in event window */
|
||||||
|
int32_t x_root, y_root; /* coordinates relative to root */
|
||||||
|
uint32_t state; /* key or button mask */
|
||||||
|
};
|
||||||
|
|
||||||
|
union InputEvent {
|
||||||
|
int type; // event type
|
||||||
|
|
||||||
|
InputEventCommon common;
|
||||||
|
|
||||||
|
// Discrete interfaces
|
||||||
|
XAnyEvent xany; // common event interface
|
||||||
|
XKeyEvent xkey; // KeyPress & KeyRelease events
|
||||||
|
XButtonEvent xbutton; // ButtonPress & ButtonRelease events
|
||||||
|
XMotionEvent xmotion; // MotionNotify event
|
||||||
|
XCrossingEvent xcrossing; // EnterNotify & LeaveNotify events
|
||||||
|
|
||||||
|
// Ensures InputEvent matches memory layout of XEvent.
|
||||||
|
// Accessing base variant is as code smell.
|
||||||
|
XEvent base;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Returns InputEvent pointer to provided XEvent is an input event; nullptr otherwise.
|
||||||
|
InputEvent *xev_as_input_event(XEvent &ev);
|
||||||
|
void propagate_x11_event(XEvent &ev);
|
||||||
|
|
||||||
|
#ifdef BUILD_MOUSE_EVENTS
|
||||||
|
|
||||||
|
Window query_x11_window_at_pos(Display* display, int x, int y);
|
||||||
|
#endif /* BUILD_MOUSE_EVENTS */
|
||||||
|
|
||||||
#ifdef BUILD_XDBE
|
#ifdef BUILD_XDBE
|
||||||
void xdbe_swap_buffers(void);
|
void xdbe_swap_buffers(void);
|
||||||
#else
|
#else
|
||||||
|
Loading…
Reference in New Issue
Block a user