mirror of
https://github.com/Llewellynvdm/conky.git
synced 2025-02-14 10:00:21 +00:00
* Introduce geometry primitives based on Vc * Apply geometry primitives to some existing code Signed-off-by: Tin Švagelj <tin.svagelj@live.com>
345 lines
9.0 KiB
C++
345 lines
9.0 KiB
C++
/*
|
|
* mouse_events.h: conky support for mouse events
|
|
*
|
|
* Copyright (C) 2020 Tin Svagelj tin.svagelj@live.com
|
|
*
|
|
* This program 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.
|
|
*
|
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#ifndef MOUSE_EVENTS_H
|
|
#define MOUSE_EVENTS_H
|
|
|
|
#include <bitset>
|
|
#include <cstdint>
|
|
#include <string>
|
|
|
|
#include "config.h"
|
|
|
|
#include "geometry.h"
|
|
#include "logging.h"
|
|
|
|
#ifdef BUILD_XINPUT
|
|
#include <array>
|
|
#include <map>
|
|
#include <optional>
|
|
#include <tuple>
|
|
#include <variant>
|
|
#include <vector>
|
|
#endif /* BUILD_XINPUT */
|
|
|
|
extern "C" {
|
|
#ifdef BUILD_X11
|
|
#include <X11/X.h>
|
|
|
|
#ifdef BUILD_XINPUT
|
|
#include <X11/extensions/XInput.h>
|
|
#include <X11/extensions/XInput2.h>
|
|
#undef COUNT // define from X11/extensions/Xi.h
|
|
|
|
#endif /* BUILD_XINPUT */
|
|
#endif /* BUILD_X11 */
|
|
|
|
#include <lua.h>
|
|
|
|
#ifdef __linux
|
|
#include <linux/input-event-codes.h>
|
|
#elif __FreeBSD__
|
|
#include <dev/evdev/input-event-codes.h>
|
|
#elif __DragonFly__
|
|
#include <dev/misc/evdev/input-event-codes.h>
|
|
#else /* platform */
|
|
// Probably incorrect for some platforms, feel free to add your platform to the
|
|
// above list if it has other event codes or a standard file containing them.
|
|
|
|
// Left mouse button event code
|
|
#define BTN_LEFT 0x110
|
|
// Right mouse button event code
|
|
#define BTN_RIGHT 0x111
|
|
// Middle mouse button event code
|
|
#define BTN_MIDDLE 0x112
|
|
|
|
// Back mouse button event code
|
|
#define BTN_BACK 0x116
|
|
// Forward mouse button event code
|
|
#define BTN_FORWARD 0x115
|
|
#endif /* platform */
|
|
}
|
|
|
|
namespace conky {
|
|
|
|
#ifdef BUILD_MOUSE_EVENTS
|
|
enum class mouse_event_t : uint32_t {
|
|
PRESS = 0,
|
|
RELEASE = 1,
|
|
SCROLL = 2,
|
|
MOVE = 3,
|
|
AREA_ENTER = 4,
|
|
AREA_LEAVE = 5,
|
|
};
|
|
const size_t MOUSE_EVENT_COUNT =
|
|
static_cast<size_t>(mouse_event_t::AREA_LEAVE) + 1;
|
|
constexpr uint32_t operator*(mouse_event_t index) {
|
|
return static_cast<uint32_t>(index);
|
|
}
|
|
|
|
enum class mouse_button_t : uint32_t {
|
|
LEFT = BTN_LEFT,
|
|
RIGHT = BTN_RIGHT,
|
|
MIDDLE = BTN_MIDDLE,
|
|
BACK = BTN_BACK,
|
|
FORWARD = BTN_FORWARD,
|
|
};
|
|
constexpr uint32_t operator*(mouse_button_t index) {
|
|
return static_cast<uint32_t>(index);
|
|
}
|
|
|
|
#ifdef BUILD_X11
|
|
inline mouse_button_t x11_mouse_button_code(unsigned int x11_mouse_button) {
|
|
mouse_button_t button;
|
|
switch (x11_mouse_button) {
|
|
case Button1:
|
|
button = mouse_button_t::LEFT;
|
|
break;
|
|
case Button2:
|
|
button = mouse_button_t::MIDDLE;
|
|
break;
|
|
case Button3:
|
|
button = mouse_button_t::RIGHT;
|
|
break;
|
|
case 8:
|
|
button = mouse_button_t::BACK;
|
|
break;
|
|
case 9:
|
|
button = mouse_button_t::FORWARD;
|
|
break;
|
|
default:
|
|
DBGP("X11 button %d is not mapped", x11_mouse_button);
|
|
break;
|
|
}
|
|
return button;
|
|
}
|
|
#endif /* BUILD_X11 */
|
|
|
|
struct mouse_event {
|
|
mouse_event_t type; // type of event
|
|
std::size_t time; // ms since epoch when the event happened
|
|
|
|
explicit mouse_event(mouse_event_t type);
|
|
|
|
void push_lua_table(lua_State *L) const;
|
|
|
|
virtual void push_lua_data(lua_State *L) const = 0;
|
|
};
|
|
|
|
struct mouse_positioned_event : public mouse_event {
|
|
/// @brief Position relative to event window
|
|
vec2<size_t> pos;
|
|
/// @brief Position relative to desktop/root window
|
|
vec2<size_t> pos_absolute;
|
|
|
|
mouse_positioned_event(mouse_event_t type, vec2<size_t> pos,
|
|
vec2<size_t> pos_absolute)
|
|
: mouse_event(type), pos(pos), pos_absolute(pos_absolute){};
|
|
|
|
void push_lua_data(lua_State *L) const;
|
|
};
|
|
|
|
typedef std::bitset<6> modifier_state_t;
|
|
enum class modifier_key : uint32_t {
|
|
SHIFT = 0,
|
|
CONTROL = 1,
|
|
ALT = 2,
|
|
// Windows/MacOS key on most keyboards
|
|
SUPER = 3,
|
|
CAPS_LOCK = 4,
|
|
NUM_LOCK = 5,
|
|
};
|
|
constexpr uint32_t operator*(modifier_key index) {
|
|
return static_cast<uint32_t>(index);
|
|
}
|
|
std::string modifier_name(modifier_key key);
|
|
|
|
#ifdef BUILD_X11
|
|
inline modifier_state_t x11_modifier_state(unsigned int mods) {
|
|
modifier_state_t result;
|
|
result[*modifier_key::SHIFT] = mods & ShiftMask;
|
|
result[*modifier_key::CONTROL] = mods & ControlMask;
|
|
result[*modifier_key::ALT] = mods & Mod1Mask;
|
|
result[*modifier_key::SUPER] = mods & Mod4Mask;
|
|
result[*modifier_key::CAPS_LOCK] = mods & LockMask;
|
|
result[*modifier_key::NUM_LOCK] = mods & Mod2Mask;
|
|
return result;
|
|
}
|
|
#endif /* BUILD_X11 */
|
|
|
|
struct mouse_move_event : public mouse_positioned_event {
|
|
modifier_state_t mods; // held buttons and modifiers (ctrl, shift, ...)
|
|
|
|
mouse_move_event(vec2<size_t> pos, vec2<size_t> pos_absolute,
|
|
modifier_state_t mods = 0)
|
|
: mouse_positioned_event{mouse_event_t::MOVE, pos, pos_absolute},
|
|
mods(mods){};
|
|
|
|
void push_lua_data(lua_State *L) const;
|
|
};
|
|
|
|
enum class scroll_direction_t : uint8_t {
|
|
UNKNOWN = 0,
|
|
UP,
|
|
DOWN,
|
|
LEFT,
|
|
RIGHT,
|
|
};
|
|
constexpr uint8_t operator*(scroll_direction_t index) {
|
|
return static_cast<uint8_t>(index);
|
|
}
|
|
|
|
#ifdef BUILD_X11
|
|
inline scroll_direction_t x11_scroll_direction(unsigned int x11_mouse_button) {
|
|
scroll_direction_t direction = scroll_direction_t::UNKNOWN;
|
|
switch (x11_mouse_button) {
|
|
case Button4:
|
|
direction = scroll_direction_t::UP;
|
|
break;
|
|
case Button5:
|
|
direction = scroll_direction_t::DOWN;
|
|
break;
|
|
case 6:
|
|
direction = scroll_direction_t::LEFT;
|
|
break;
|
|
case 7:
|
|
direction = scroll_direction_t::RIGHT;
|
|
break;
|
|
}
|
|
return direction;
|
|
}
|
|
#endif /* BUILD_X11 */
|
|
|
|
struct mouse_scroll_event : public mouse_positioned_event {
|
|
modifier_state_t mods; // held buttons and modifiers (ctrl, shift, ...)
|
|
scroll_direction_t direction;
|
|
|
|
mouse_scroll_event(vec2<size_t> pos, vec2<size_t> pos_absolute,
|
|
scroll_direction_t direction, modifier_state_t mods = 0)
|
|
: mouse_positioned_event{mouse_event_t::SCROLL, pos, pos_absolute},
|
|
direction(direction),
|
|
mods(mods){};
|
|
|
|
void push_lua_data(lua_State *L) const;
|
|
};
|
|
|
|
struct mouse_button_event : public mouse_positioned_event {
|
|
modifier_state_t mods; // held buttons and modifiers (ctrl, shift, ...)
|
|
mouse_button_t button;
|
|
|
|
mouse_button_event(mouse_event_t type, vec2<size_t> pos,
|
|
vec2<size_t> pos_absolute, mouse_button_t button,
|
|
modifier_state_t mods = 0)
|
|
: mouse_positioned_event{type, pos, pos_absolute},
|
|
button(button),
|
|
mods(mods){};
|
|
|
|
void push_lua_data(lua_State *L) const;
|
|
};
|
|
|
|
struct mouse_crossing_event : public mouse_positioned_event {
|
|
mouse_crossing_event(mouse_event_t type, vec2<size_t> pos,
|
|
vec2<size_t> pos_absolute)
|
|
: mouse_positioned_event{type, pos, pos_absolute} {};
|
|
};
|
|
#endif /* BUILD_MOUSE_EVENTS */
|
|
|
|
#ifdef BUILD_XINPUT
|
|
typedef int xi_device_id;
|
|
typedef int xi_event_type;
|
|
|
|
enum class valuator_t : size_t { MOVE_X, MOVE_Y, SCROLL_X, SCROLL_Y, UNKNOWN };
|
|
const size_t VALUATOR_COUNT = static_cast<size_t>(valuator_t::UNKNOWN);
|
|
constexpr uint8_t operator*(valuator_t index) {
|
|
return static_cast<uint8_t>(index);
|
|
}
|
|
|
|
struct conky_valuator_info {
|
|
size_t index;
|
|
double min;
|
|
double max;
|
|
double value;
|
|
bool relative;
|
|
};
|
|
|
|
struct device_info {
|
|
/// @brief Device name.
|
|
xi_device_id id;
|
|
std::string name;
|
|
std::array<conky_valuator_info, VALUATOR_COUNT> valuators{};
|
|
|
|
static device_info *from_xi_id(xi_device_id id, Display *display = nullptr);
|
|
|
|
conky_valuator_info &valuator(valuator_t valuator);
|
|
|
|
private:
|
|
void init_xi_device(Display *display,
|
|
std::variant<xi_device_id, XIDeviceInfo *> device);
|
|
};
|
|
|
|
void handle_xi_device_change(const XIHierarchyEvent *event);
|
|
|
|
/// Almost an exact copy of `XIDeviceEvent`, except it owns all data.
|
|
struct xi_event_data {
|
|
xi_event_type evtype;
|
|
unsigned long serial;
|
|
Bool send_event;
|
|
Display *display;
|
|
/// XI extension offset
|
|
// TODO: Check whether this is consistent between different clients by
|
|
// printing.
|
|
int extension;
|
|
Time time;
|
|
device_info *device;
|
|
int sourceid;
|
|
int detail;
|
|
Window root;
|
|
Window event;
|
|
Window child;
|
|
conky::vec2d pos_absolute;
|
|
conky::vec2d pos;
|
|
int flags;
|
|
/// pressed button mask
|
|
std::bitset<32> buttons;
|
|
std::map<size_t, double> valuators;
|
|
XIModifierState mods;
|
|
XIGroupState group;
|
|
|
|
// Extra data
|
|
|
|
/// Precomputed relative values
|
|
std::array<double, VALUATOR_COUNT> valuators_relative;
|
|
|
|
static xi_event_data *read_cookie(Display *display, const void *data);
|
|
|
|
bool test_valuator(valuator_t id) const;
|
|
conky_valuator_info *valuator_info(valuator_t id) const;
|
|
std::optional<double> valuator_value(valuator_t id) const;
|
|
std::optional<double> valuator_relative_value(valuator_t valuator) const;
|
|
|
|
std::vector<std::tuple<int, XEvent *>> generate_events(
|
|
Window target, Window child, conky::vec2d target_pos) const;
|
|
};
|
|
|
|
#endif /* BUILD_XINPUT */
|
|
} // namespace conky
|
|
|
|
#endif /* MOUSE_EVENTS_H */
|