mirror of
https://github.com/WayfireWM/wf-osk.git
synced 2025-04-05 12:56:45 +02:00
200 lines
5.9 KiB
C++
200 lines
5.9 KiB
C++
#include "osk.hpp"
|
|
#include <getopt.h>
|
|
#include <iostream>
|
|
#include <linux/input-event-codes.h>
|
|
|
|
#include "util/clara.hpp"
|
|
|
|
#define ABC_TOGGLE 0x12345678
|
|
#define NUM_TOGGLE 0x87654321
|
|
|
|
#define IS_COMMAND(x) ((x) == ABC_TOGGLE || (x) == NUM_TOGGLE)
|
|
|
|
#define USE_SHIFT 0x10000000
|
|
|
|
namespace wf
|
|
{
|
|
namespace osk
|
|
{
|
|
int spacing = OSK_SPACING;
|
|
int default_width = 800;
|
|
int default_height = 400;
|
|
std::string anchor;
|
|
|
|
KeyButton::KeyButton(Key key, int width, int height)
|
|
{
|
|
this->code = key.code;
|
|
|
|
this->button.set_size_request(width, height);
|
|
this->button.set_label(key.text);
|
|
|
|
this->button.signal_pressed().connect_notify(
|
|
sigc::mem_fun(this, &KeyButton::on_pressed));
|
|
this->button.signal_released().connect_notify(
|
|
sigc::mem_fun(this, &KeyButton::on_released));
|
|
}
|
|
|
|
void KeyButton::on_pressed()
|
|
{
|
|
auto& keyboard = Keyboard::get();
|
|
if (IS_COMMAND(this->code))
|
|
return;
|
|
|
|
if (this->code & USE_SHIFT)
|
|
keyboard.get_device().set_shift(true);
|
|
|
|
keyboard.get_device().send_key(this->code & ~(USE_SHIFT),
|
|
WL_KEYBOARD_KEY_STATE_PRESSED);
|
|
}
|
|
|
|
void KeyButton::on_released()
|
|
{
|
|
auto& keyboard = Keyboard::get();
|
|
if (IS_COMMAND(this->code))
|
|
return keyboard.handle_action(this->code);
|
|
|
|
if (this->code & USE_SHIFT)
|
|
keyboard.get_device().set_shift(false);
|
|
|
|
keyboard.get_device().send_key(this->code & ~(USE_SHIFT),
|
|
WL_KEYBOARD_KEY_STATE_RELEASED);
|
|
}
|
|
|
|
KeyboardRow::KeyboardRow(std::vector<Key> keys,
|
|
int width, int height)
|
|
{
|
|
double sum = 0;
|
|
for (auto& key : keys)
|
|
sum += key.width;
|
|
|
|
box.set_spacing(spacing);
|
|
int total_spacing = std::min((int)keys.size() - 1, 0) * spacing;
|
|
int total_buttons = width - total_spacing;
|
|
|
|
for (auto& key : keys)
|
|
{
|
|
this->keys.emplace_back(std::make_unique<KeyButton>
|
|
(key, int(key.width / sum * total_buttons), height));
|
|
this->box.pack_start(this->keys.back()->button);
|
|
}
|
|
}
|
|
|
|
KeyboardLayout::KeyboardLayout(std::vector<std::vector<Key>> keys,
|
|
int32_t width, int32_t height)
|
|
{
|
|
box.set_spacing(spacing);
|
|
int total_spacing = std::min((int)keys.size() - 1, 0) * spacing;
|
|
|
|
int row_height = (height - total_spacing) / keys.size();
|
|
for (auto& row : keys)
|
|
{
|
|
this->rows.emplace_back(
|
|
std::make_unique<KeyboardRow> (row, width, row_height));
|
|
this->box.pack_start(this->rows.back()->box);
|
|
}
|
|
}
|
|
|
|
void Keyboard::init_layouts()
|
|
{
|
|
/* Key layouts are defined in layouts.tpp,
|
|
* it defines default_keys, shift_keys, numeric_keys */
|
|
#include "layouts.tpp"
|
|
|
|
this->default_layout = std::make_unique<KeyboardLayout>
|
|
(default_keys, default_width, default_height);
|
|
|
|
this->shift_layout = std::make_unique<KeyboardLayout>
|
|
(shift_keys, default_width, default_height);
|
|
|
|
this->numeric_layout = std::make_unique<KeyboardLayout>
|
|
(numeric_keys, default_width, default_height);
|
|
}
|
|
|
|
void Keyboard::set_layout(KeyboardLayout *new_layout)
|
|
{
|
|
this->current_layout = new_layout;
|
|
window->set_widget(new_layout->box);
|
|
}
|
|
|
|
Keyboard::Keyboard()
|
|
{
|
|
window = std::make_unique<WaylandWindow>
|
|
(default_width, default_height, anchor);
|
|
vk = std::make_unique<VirtualKeyboardDevice> ();
|
|
|
|
init_layouts();
|
|
set_layout(default_layout.get());
|
|
}
|
|
|
|
std::unique_ptr<Keyboard> Keyboard::instance;
|
|
void Keyboard::create()
|
|
{
|
|
if (instance)
|
|
throw std::logic_error("Creating keyboard twice!");
|
|
|
|
instance = std::unique_ptr<Keyboard>(new Keyboard());
|
|
}
|
|
|
|
Keyboard& Keyboard::get()
|
|
{
|
|
if (!instance)
|
|
throw std::logic_error("Getting keyboard before creating it!");
|
|
|
|
return *instance;
|
|
}
|
|
|
|
VirtualKeyboardDevice& Keyboard::get_device()
|
|
{
|
|
return *vk;
|
|
}
|
|
|
|
Gtk::Window& Keyboard::get_window()
|
|
{
|
|
return *window;
|
|
}
|
|
|
|
void Keyboard::handle_action(uint32_t action)
|
|
{
|
|
if (action == ABC_TOGGLE)
|
|
{
|
|
if (current_layout == default_layout.get()) {
|
|
set_layout(shift_layout.get());
|
|
} else {
|
|
set_layout(default_layout.get());
|
|
}
|
|
}
|
|
|
|
if (action == NUM_TOGGLE)
|
|
set_layout(numeric_layout.get());
|
|
}
|
|
}
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
bool show_help = false;
|
|
|
|
auto cli = clara::detail::Help(show_help) |
|
|
clara::detail::Opt(wf::osk::default_width, "int")["-w"]["--width"]
|
|
("keyboard width") |
|
|
clara::detail::Opt(wf::osk::default_height, "int")["-h"]["--height"]
|
|
("keyboard height") |
|
|
clara::detail::Opt(wf::osk::anchor, "top|left|bottom|right")["-a"]
|
|
["--anchor"]("where the keyboard should anchor in the screen");
|
|
|
|
auto res = cli.parse(clara::detail::Args(argc, argv));
|
|
if (!res) {
|
|
std::cerr << "Error: " << res.errorMessage() << std::endl;
|
|
return 1;
|
|
}
|
|
|
|
if (show_help) {
|
|
std::cout << cli << std::endl;
|
|
return 0;
|
|
}
|
|
|
|
auto app = Gtk::Application::create();
|
|
wf::osk::Keyboard::create();
|
|
return app->run(wf::osk::Keyboard::get().get_window());
|
|
}
|