-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindev.c
197 lines (160 loc) · 6.21 KB
/
indev.c
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
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
/**
* Copyright 2021 Johannes Marbach
*
* This file is part of bootman, hereafter referred to as the program.
*
* 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/>.
*/
#include "indev.h"
#include "cursor.h"
#include "lv_drivers/indev/libinput_drv.h"
#include <limits.h>
#include <stdio.h>
/**
* Defines
*/
#define MAX_KEYBOARD_DEVS 4
#define MAX_POINTER_DEVS 4
#define MAX_TOUCHSCREEN_DEVS 2
/**
* Static variables
*/
static int num_keyboard_devs = 0;
static char *keyboard_devs[MAX_KEYBOARD_DEVS];
static lv_indev_t *keyboard_indevs[MAX_KEYBOARD_DEVS];
static lv_indev_drv_t keyboard_indev_drvs[MAX_KEYBOARD_DEVS];
static libinput_drv_state_t keyboard_drv_states[MAX_KEYBOARD_DEVS];
static int num_pointer_devs = 0;
static char *pointer_devs[MAX_POINTER_DEVS];
static lv_indev_t *pointer_indevs[MAX_POINTER_DEVS];
static lv_indev_drv_t pointer_indev_drvs[MAX_POINTER_DEVS];
static libinput_drv_state_t pointer_drv_states[MAX_POINTER_DEVS];
static int num_touchscreen_devs = 0;
static char *touchscreen_devs[MAX_TOUCHSCREEN_DEVS];
static lv_indev_t *touchscreen_indevs[MAX_TOUCHSCREEN_DEVS];
static lv_indev_drv_t touchscreen_indev_drvs[MAX_TOUCHSCREEN_DEVS];
static libinput_drv_state_t touchscreen_drv_states[MAX_TOUCHSCREEN_DEVS];
/**
* Static prototypes
*/
/**
* Auto-connect available input devices having a specific capability.
*
* @param capability capability to filter devices by
* @param max_num_devs maximum number of devices to connect
* @param num_devs pointer for writing the actual number of connected devices into
* @param devs array for storing device paths
* @param indevs array for storing LVGL indevs
* @param indev_drvs array for storing LVGL indev drivers
* @param drv_states array for storing LVGL libinput driver states
*/
static void auto_connect(libinput_capability capability, int max_num_devs, int *num_devs, char *devs[], lv_indev_t *indevs[],
lv_indev_drv_t indev_drvs[], libinput_drv_state_t drv_states[]);
/**
* Log a message announcing the connection of an input device.
*
* @param capability the device's capability
* @param dev the device path
*/
static void log_connection(libinput_capability capability, char *dev);
/**
* Perform an input read on a device using the libinput driver.
*
* @param indev_drv input device driver
* @param data input device data to write into
*/
static void libinput_read_cb(lv_indev_drv_t *indev_drv, lv_indev_data_t *data);
/**
* Static functions
*/
static void auto_connect(libinput_capability capability, int max_num_devs, int *num_devs, char *devs[], lv_indev_t *indevs[],
lv_indev_drv_t indev_drvs[], libinput_drv_state_t drv_states[]) {
memset(devs, 0, max_num_devs * sizeof(char *));
memset(indevs, 0, max_num_devs * sizeof(lv_indev_t *));
memset(indev_drvs, 0, max_num_devs * sizeof(lv_indev_drv_t));
memset(drv_states, 0, max_num_devs * sizeof(libinput_drv_state_t));
*num_devs = libinput_find_devs(capability, devs, max_num_devs, false);
for (int i = 0; i < *num_devs; ++i) {
log_connection(capability, devs[i]);
libinput_init_state(&(drv_states[i]), devs[i]);
lv_indev_drv_init(&(indev_drvs[i]));
indev_drvs[i].read_cb = libinput_read_cb;
indev_drvs[i].user_data = &(drv_states[i]);
if (capability == LIBINPUT_CAPABILITY_KEYBOARD) {
indev_drvs[i].type = LV_INDEV_TYPE_KEYPAD;
} else {
indev_drvs[i].type = LV_INDEV_TYPE_POINTER;
indev_drvs[i].long_press_repeat_time = USHRT_MAX;
}
indevs[i] = lv_indev_drv_register(&(indev_drvs[i]));
}
}
static void log_connection(libinput_capability capability, char *dev) {
switch (capability) {
case LIBINPUT_CAPABILITY_KEYBOARD:
printf("Connecting keyboard device %s\n", dev);
break;
case LIBINPUT_CAPABILITY_POINTER:
printf("Connecting pointer device %s\n", dev);
break;
case LIBINPUT_CAPABILITY_TOUCH:
printf("Connecting touchscreen device %s\n", dev);
break;
case LIBINPUT_CAPABILITY_NONE:
break;
}
}
static void libinput_read_cb(lv_indev_drv_t *indev_drv, lv_indev_data_t *data) {
libinput_read_state(indev_drv->user_data, indev_drv, data);
}
/**
* Public functions
*/
void indev_auto_connect(bool keyboard, bool pointer, bool touchscreen) {
if (keyboard) {
auto_connect(LIBINPUT_CAPABILITY_KEYBOARD, MAX_KEYBOARD_DEVS, &num_keyboard_devs, keyboard_devs, keyboard_indevs,
keyboard_indev_drvs, keyboard_drv_states);
}
if (pointer) {
auto_connect(LIBINPUT_CAPABILITY_POINTER, MAX_POINTER_DEVS, &num_pointer_devs, pointer_devs, pointer_indevs,
pointer_indev_drvs, pointer_drv_states);
}
if (touchscreen) {
auto_connect(LIBINPUT_CAPABILITY_TOUCH, MAX_TOUCHSCREEN_DEVS, &num_touchscreen_devs, touchscreen_devs, touchscreen_indevs,
touchscreen_indev_drvs, touchscreen_drv_states);
}
}
bool indev_is_keyboard_connected() {
return num_keyboard_devs > 0;
}
void indev_set_up_textarea_for_keyboard_input(lv_obj_t *textarea) {
if (!indev_is_keyboard_connected()) {
return;
}
lv_group_t *group = lv_group_create();
lv_group_add_obj(group, textarea);
for (int i = 0; i < num_keyboard_devs; ++i) {
lv_indev_set_group(keyboard_indevs[i], group);
}
}
void indev_set_up_mouse_cursor() {
if (num_pointer_devs == 0) {
return;
}
lv_obj_t *cursor_obj = lv_img_create(lv_scr_act());
lv_img_set_src(cursor_obj, &cursor_img_dsc);
for (int i = 0; i < num_pointer_devs; ++i) {
lv_indev_set_cursor(pointer_indevs[i], cursor_obj);
}
}