Compare commits

...

25 Commits

Author SHA1 Message Date
Francisco Lopes
6ad0031138
Mention Void Linux install
Closes #22
2021-08-22 09:28:44 -03:00
Francisco Lopes
56ea92caa2
Update README 2021-05-10 23:01:58 -03:00
Francisco Lopes
4f1abaf937
Add install step for Ubuntu 2021-04-02 13:03:17 -03:00
Francisco Lopes
1d92f46e4e
Update README 2021-01-13 10:57:03 -03:00
Francisco Lopes
dd7cb3e467
Remove Ubuntu notice to let it open to contributions 2021-01-13 10:54:51 -03:00
Francisco Lopes
aae8117eeb
"usage" uniformity 2021-01-07 16:07:59 -03:00
Francisco Lopes
b99ee5b6be
Update README 2021-01-07 02:30:53 -03:00
Francisco Lopes
3c0b816499
Update README 2021-01-07 02:29:20 -03:00
Francisco Lopes
be414120a2
Update README 2021-01-07 02:24:06 -03:00
Francisco Lopes
5cffbf6bb5
Formatting 2021-01-02 21:42:05 -03:00
Francisco Lopes
9aa9985b0e
Update README 2021-01-01 11:58:39 -03:00
Francisco Lopes
466062ef00
Update README 2021-01-01 07:44:58 -03:00
Francisco Lopes
975bb74284
Passthrough mouse events
To not block mice that have keys, which end up grabbed due to generic
configurations that simply filter anything that responds to CAPSLOCK/ESC.

Fixes #15
2021-01-01 00:18:03 -03:00
Francisco Lopes
8bfd57eda2
Formatting 2020-12-30 23:40:02 -03:00
Francisco Lopes
186b887e7d
Make multi device example more robust
The previous one won't work when devices reattach.
2020-12-30 23:16:17 -03:00
Francisco Lopes
c738e8783a
Formatting 2020-12-30 21:32:58 -03:00
Francisco Lopes
a062311ffb
Formatting 2020-12-30 21:19:31 -03:00
Francisco Lopes
ac6bbe61c2
Update README 2020-12-29 09:26:35 -03:00
Francisco Lopes
12dc2f75c7
Update README 2020-12-29 03:47:41 -03:00
Francisco Lopes
3eaf2bddfe
Update README 2020-12-29 03:46:32 -03:00
Francisco Lopes
0c8326646a
Expand mouse support
Until now mouse support existed just as side effect of muxing and that
mouse clicks are registered as key events. This expands the support for
mouse wheel and movement.
2020-12-29 02:43:04 -03:00
Francisco Lopes
5c7e6b5a5d
Formatting 2020-12-28 23:35:47 -03:00
Francisco Lopes
7428d959e0
Formatting 2020-12-28 23:31:58 -03:00
Francisco Lopes
e5dc581ddf
Formatting 2020-12-28 23:24:11 -03:00
Francisco Lopes
41a2fa7f1e
Rewrite in state machine look
- Also add new option to control internal delay of programmatic key
  sequences, when they're necessary.
2020-12-28 23:13:34 -03:00
2 changed files with 123 additions and 105 deletions

View File

@ -25,8 +25,8 @@ ESC when pressed alone is quite handy, specially in vi.
## Building ## Building
``` ```text
$ git clone git@gitlab.com:interception/linux/plugins/caps2esc.git $ git clone https://gitlab.com/interception/linux/plugins/caps2esc.git
$ cd caps2esc $ cd caps2esc
$ cmake -B build -DCMAKE_BUILD_TYPE=Release $ cmake -B build -DCMAKE_BUILD_TYPE=Release
$ cmake --build build $ cmake --build build
@ -34,75 +34,87 @@ $ cmake --build build
## Execution ## Execution
``` ```text
caps2esc - transforming the most useless key ever in the most useful one caps2esc - transforming the most useless key ever in the most useful one
usage: caps2esc [-h] [-m mode] usage: caps2esc [-h | [-m mode] [-t delay]]
options: options:
-h show this message and exit -h show this message and exit
-t delay used for key sequences (default: 20000 microseconds)
-m mode 0: default -m mode 0: default
- caps as esc/ctrl - caps as esc/ctrl
- esc as caps - esc as caps
1: minimal 1: minimal
- caps as esc/ctrl - caps as esc/ctrl
2: useful on 60% layouts 2: useful on 60% layouts
- caps as esc/ctrl - caps as esc/ctrl
- esc as grave accent - esc as grave accent
- grave accent as caps - grave accent as caps
``` ```
`caps2esc` is an [_Interception Tools_][interception-tools] plugin. A suggested `caps2esc` is an [_Interception Tools_][interception-tools] plugin. A suggested
`udevmon` job configuration is: `udevmon` job configuration (check the [_Interception Tools_
README][interception-tools] for alternatives) is:
```yaml ```yaml
- JOB: intercept -g $DEVNODE | caps2esc | uinput -d $DEVNODE - JOB: intercept -g $DEVNODE | caps2esc | uinput -d $DEVNODE
DEVICE: DEVICE:
EVENTS: EVENTS:
EV_KEY: [KEY_CAPSLOCK, KEY_ESC] EV_KEY: [KEY_CAPSLOCK, KEY_ESC]
``` ```
For more information about the [_Interception Tools_][interception-tools], check For more information about the [_Interception Tools_][interception-tools], check
the project's website. the project's website.
## Mouse button support ## Mouse/Touchpad Support
After _Interception Tools_ 0.3, `caps2esc` can work with mouse clicks. An After _Interception Tools_ 0.3.2, `caps2esc` can observe (or replace) mouse
example configuration taken from my laptop: events. An example configuration taken from my laptop:
```yaml ```yaml
SHELL: [zsh, -c] SHELL: [zsh, -c]
--- ---
- JOB: mux -c caps2esc - CMD: mux -c caps2esc
- JOB: - JOB: mux -i caps2esc | caps2esc | uinput -c /etc/interception/keyboard.yaml
- intercept -g $DEVNODE | mux -o caps2esc - JOB: intercept -g $DEVNODE | mux -o caps2esc
- mux -i caps2esc | caps2esc | uinput -d $DEVNODE
DEVICE: DEVICE:
LINK: /dev/input/by-path/platform-i8042-serio-0-event-kbd LINK: /dev/input/by-path/platform-i8042-serio-0-event-kbd
- JOB: - JOB: intercept $DEVNODE | mux -o caps2esc
- intercept $DEVNODE | mux -o caps2esc
DEVICE: DEVICE:
LINK: /dev/input/by-path/platform-i8042-serio-4-event-mouse LINK: /dev/input/by-path/platform-i8042-serio-4-event-mouse
``` ```
For more information on the topic, check the [_Interception Tools_ For more information on the topic, check the [_Interception Tools_
README][interception-tools] about usage of the `mux` tool and device specific README][interception-tools] about usage of the `mux` tool and hybrid virtual
setups. device configurations.
## Installation ## Installation
### Archlinux
It's available from [community](https://archlinux.org/packages/community/x86_64/interception-caps2esc/): It's available from [community](https://archlinux.org/packages/community/x86_64/interception-caps2esc/):
``` ```text
$ pacman -S interception-caps2esc $ pacman -S interception-caps2esc
``` ```
I don't use Ubuntu and recommend Archlinux instead, as it provides the AUR, so I ### Void Linux
don't maintain PPAs. For more information on Ubuntu/Debian installation check
this:
- <https://askubuntu.com/questions/979359/how-do-i-install-caps2esc> ```text
$ xbps-install -S caps2esc
```
### Ubuntu ([independent package][ubuntu])
```text
sudo add-apt-repository ppa:deafmute/interception
sudo apt install interception-caps2esc
```
<sub>For debian and other derivatives you can download directly at https://launchpad.net/~deafmute/+archive/ubuntu/interception/+packages.</sub>
[ubuntu]: https://gitlab.com/interception/linux/tools/-/issues/38
## Caveats ## Caveats
@ -117,7 +129,7 @@ As always, there's always a caveat:
I can't recall when I started using CAPSLOCK as both ESC and CTRL but it has I can't recall when I started using CAPSLOCK as both ESC and CTRL but it has
been quite some time already. It started when I was on OS X where it was quite been quite some time already. It started when I was on OS X where it was quite
easy to achieve using the [Karabiner][], which already provides an option to easy to achieve using the [Karabiner][], which already provides an option to
turn CTRL into CTRL/ESC (which can be coupled with OS X system settings that turn CTRL into ESC/CTRL (which can be coupled with OS X system settings that
turn CAPSLOCK into CTRL). turn CAPSLOCK into CTRL).
Moving on, permanently making Linux my home, I searched and tweaked a similar Moving on, permanently making Linux my home, I searched and tweaked a similar

View File

@ -6,16 +6,11 @@
// clang-format off // clang-format off
const struct input_event const struct input_event
esc_up = {.type = EV_KEY, .code = KEY_ESC, .value = 0}, syn = {.type = EV_SYN , .code = SYN_REPORT , .value = 0},
ctrl_up = {.type = EV_KEY, .code = KEY_LEFTCTRL, .value = 0}, esc_up = {.type = EV_KEY , .code = KEY_ESC , .value = 0},
capslock_up = {.type = EV_KEY, .code = KEY_CAPSLOCK, .value = 0}, ctrl_up = {.type = EV_KEY , .code = KEY_LEFTCTRL , .value = 0},
esc_down = {.type = EV_KEY, .code = KEY_ESC, .value = 1}, esc_down = {.type = EV_KEY , .code = KEY_ESC , .value = 1},
ctrl_down = {.type = EV_KEY, .code = KEY_LEFTCTRL, .value = 1}, ctrl_down = {.type = EV_KEY , .code = KEY_LEFTCTRL , .value = 1};
capslock_down = {.type = EV_KEY, .code = KEY_CAPSLOCK, .value = 1},
esc_repeat = {.type = EV_KEY, .code = KEY_ESC, .value = 2},
ctrl_repeat = {.type = EV_KEY, .code = KEY_LEFTCTRL, .value = 2},
capslock_repeat = {.type = EV_KEY, .code = KEY_CAPSLOCK, .value = 2},
syn = {.type = EV_SYN, .code = SYN_REPORT, .value = 0};
// clang-format on // clang-format on
void print_usage(FILE *stream, const char *program) { void print_usage(FILE *stream, const char *program) {
@ -23,28 +18,24 @@ void print_usage(FILE *stream, const char *program) {
fprintf(stream, fprintf(stream,
"caps2esc - transforming the most useless key ever in the most useful one\n" "caps2esc - transforming the most useless key ever in the most useful one\n"
"\n" "\n"
"usage: %s [-h] [-m mode]\n" "usage: %s [-h | [-m mode] [-t delay]]\n"
"\n" "\n"
"options:\n" "options:\n"
" -h show this message and exit\n" " -h show this message and exit\n"
" -m mode 0: default\n" " -t delay used for key sequences (default: 20000 microseconds)\n"
" - caps as esc/ctrl\n" " -m mode 0: default\n"
" - esc as caps\n" " - caps as esc/ctrl\n"
" 1: minimal\n" " - esc as caps\n"
" - caps as esc/ctrl\n" " 1: minimal\n"
" 2: useful on 60%% layouts\n" " - caps as esc/ctrl\n"
" - caps as esc/ctrl\n" " 2: useful on 60%% layouts\n"
" - esc as grave accent\n" " - caps as esc/ctrl\n"
" - grave accent as caps\n", " - esc as grave accent\n"
" - grave accent as caps\n",
program); program);
// clang-format on // clang-format on
} }
int equal(const struct input_event *first, const struct input_event *second) {
return first->type == second->type && first->code == second->code &&
first->value == second->value;
}
int read_event(struct input_event *event) { int read_event(struct input_event *event) {
return fread(event, sizeof(struct input_event), 1, stdin) == 1; return fread(event, sizeof(struct input_event), 1, stdin) == 1;
} }
@ -54,22 +45,47 @@ void write_event(const struct input_event *event) {
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
void write_event_with_mode(struct input_event *event, int mode) {
if (event->type == EV_KEY)
switch (mode) {
case 0:
if (event->code == KEY_ESC)
event->code = KEY_CAPSLOCK;
break;
case 2:
switch (event->code) {
case KEY_ESC:
event->code = KEY_GRAVE;
break;
case KEY_GRAVE:
event->code = KEY_CAPSLOCK;
break;
}
break;
}
write_event(event);
}
int main(int argc, char *argv[]) { int main(int argc, char *argv[]) {
int opt, mode = 0; int mode = 0, delay = 20000;
while ((opt = getopt(argc, argv, "hm:")) != -1) {
for (int opt; (opt = getopt(argc, argv, "ht:m:")) != -1;) {
switch (opt) { switch (opt) {
case 'h': case 'h':
return print_usage(stdout, argv[0]), EXIT_SUCCESS; return print_usage(stdout, argv[0]), EXIT_SUCCESS;
case 'm': case 'm':
mode = optarg[0] - '0'; mode = atoi(optarg);
continue;
case 't':
delay = atoi(optarg);
continue; continue;
} }
return print_usage(stderr, argv[0]), EXIT_FAILURE; return print_usage(stderr, argv[0]), EXIT_FAILURE;
} }
int capslock_is_down = 0, esc_give_up = 0;
struct input_event input; struct input_event input;
enum { START, CAPSLOCK_HELD, CAPSLOCK_IS_CTRL } state = START;
setbuf(stdin, NULL), setbuf(stdout, NULL); setbuf(stdin, NULL), setbuf(stdout, NULL);
@ -77,58 +93,48 @@ int main(int argc, char *argv[]) {
if (input.type == EV_MSC && input.code == MSC_SCAN) if (input.type == EV_MSC && input.code == MSC_SCAN)
continue; continue;
if (input.type != EV_KEY) { if (input.type != EV_KEY && input.type != EV_REL &&
input.type != EV_ABS) {
write_event(&input); write_event(&input);
continue; continue;
} }
if (capslock_is_down) { switch (state) {
if (equal(&input, &capslock_down) || case START:
equal(&input, &capslock_repeat)) if (input.type == EV_KEY && input.code == KEY_CAPSLOCK &&
continue; input.value)
state = CAPSLOCK_HELD;
if (equal(&input, &capslock_up)) { else
capslock_is_down = 0; write_event_with_mode(&input, mode);
if (esc_give_up) {
esc_give_up = 0;
write_event(&ctrl_up);
continue;
}
write_event(&esc_down);
write_event(&syn);
usleep(20000);
write_event(&esc_up);
continue;
}
if (!esc_give_up && input.value) {
esc_give_up = 1;
write_event(&ctrl_down);
write_event(&syn);
usleep(20000);
}
} else if (equal(&input, &capslock_down)) {
capslock_is_down = 1;
continue;
}
switch (mode) {
case 0:
if (input.code == KEY_ESC)
input.code = KEY_CAPSLOCK;
break; break;
case 2: case CAPSLOCK_HELD:
switch (input.code) { if (input.type == EV_KEY && input.code == KEY_CAPSLOCK) {
case KEY_ESC: if (input.value == 0) {
input.code = KEY_GRAVE; write_event(&esc_down);
break; write_event(&syn);
case KEY_GRAVE: usleep(delay);
input.code = KEY_CAPSLOCK; write_event(&esc_up);
break; state = START;
} }
} else if ((input.type == EV_KEY && input.value == 1) ||
input.type == EV_REL || input.type == EV_ABS) {
write_event(&ctrl_down);
write_event(&syn);
usleep(delay);
write_event_with_mode(&input, mode);
state = CAPSLOCK_IS_CTRL;
} else
write_event_with_mode(&input, mode);
break;
case CAPSLOCK_IS_CTRL:
if (input.type == EV_KEY && input.code == KEY_CAPSLOCK) {
input.code = KEY_LEFTCTRL;
write_event(&input);
if (input.value == 0)
state = START;
} else
write_event_with_mode(&input, mode);
break; break;
} }
write_event(&input);
} }
} }