1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
|
/* Copyright (c) 2015-2016, Stanislaw Halik <sthalik@misaki.pl>
* Permission to use, copy, modify, and/or distribute this
* software for any purpose with or without fee is hereby granted,
* provided that the above copyright notice and this permission
* notice appear in all copies.
*/
#pragma once
#include "dinput.hpp"
#include "compat/timer.hpp"
#include "compat/qhash.hpp"
#include "export.hpp"
#include <memory>
#include <vector>
#include <functional>
#include <unordered_map>
#include <iterator>
#include <QString>
#include <QMutex>
namespace win32_joy_impl {
static constexpr inline unsigned max_buttons = std::size(DIJOYSTATE2().rgbButtons);
static constexpr inline unsigned max_pov_hats = std::size(DIJOYSTATE2().rgdwPOV);
static constexpr inline unsigned pov_hat_directions = 8;
// cf. https://docs.microsoft.com/en-us/previous-versions/windows/desktop/ee416628(v=vs.85)
// see also remarks on the page
// no need to check for pos == unsigned(-1) || pos == 0xffff,
// this logic doesn't require that
static constexpr inline unsigned value_per_pov_hat_direction = 36000 / pov_hat_directions;
static constexpr inline unsigned max_buttons_and_pov_hats = max_buttons + max_pov_hats * pov_hat_directions;
//static_assert(pov_hat_directions == 4 || pov_hat_directions == 8);
// XXX how many axis update events can we reasonably get in a short time frame?
static constexpr inline unsigned num_buffers = 16;
//#define WIN32_JOY_DEBUG
struct OTR_DINPUT_EXPORT win32_joy_ctx final
{
using fn = std::function<void(const QString& guid, int btn, bool held)>;
struct joy final
{
LPDIRECTINPUTDEVICE8 joy_handle;
QString guid, name;
bool last_state[max_buttons_and_pov_hats] {};
Timer first_timer;
static DIDEVICEOBJECTDATA keystate_buffers[num_buffers];
joy(LPDIRECTINPUTDEVICE8 handle, const QString& guid, const QString& name);
~joy();
void release();
bool poll(fn const& f);
};
using joys_t = std::unordered_map<QString, std::shared_ptr<joy>>;
static constexpr inline int joy_axis_size = 65536;
struct joy_info
{
QString name, guid;
};
void poll(fn const& f);
bool poll_axis(const QString& guid, int* axes);
std::vector<joy_info> get_joy_info();
win32_joy_ctx(const win32_joy_ctx&) = delete;
win32_joy_ctx& operator=(const win32_joy_ctx&) = delete;
win32_joy_ctx();
void refresh();
private:
static QString guid_to_string(const GUID& guid);
class OTR_DINPUT_EXPORT enum_state final
{
std::vector<QString> all;
joys_t joys;
di_t di {};
static BOOL CALLBACK EnumJoysticksCallback(const DIDEVICEINSTANCE* pdidInstance, VOID* pContext);
static BOOL CALLBACK EnumObjectsCallback(const DIDEVICEOBJECTINSTANCE* pdidoi, VOID* ctx);
public:
static QMutex mtx;
enum_state();
~enum_state();
void refresh();
const joys_t& get_joys() const;
enum_state(enum_state const&) = delete;
};
static enum_state enumerator;
};
} // ns win32_joy_impl
using win32_joy_ctx = win32_joy_impl::win32_joy_ctx;
|