mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-11-23 13:03:13 +00:00
214 lines
6.4 KiB
C
214 lines
6.4 KiB
C
#include "hid_mouse_clicker.h"
|
|
#include <gui/elements.h>
|
|
#include "../hid.h"
|
|
|
|
#include "hid_icons.h"
|
|
|
|
#define TAG "HidMouseClicker"
|
|
#define DEFAULT_CLICK_RATE 1
|
|
#define MAXIMUM_CLICK_RATE 60
|
|
|
|
struct HidMouseClicker {
|
|
View* view;
|
|
Hid* hid;
|
|
FuriTimer* timer;
|
|
};
|
|
|
|
typedef struct {
|
|
bool connected;
|
|
bool running;
|
|
int rate;
|
|
HidTransport transport;
|
|
} HidMouseClickerModel;
|
|
|
|
static void hid_mouse_clicker_start_or_restart_timer(void* context) {
|
|
furi_assert(context);
|
|
HidMouseClicker* hid_mouse_clicker = context;
|
|
|
|
if(furi_timer_is_running(hid_mouse_clicker->timer)) {
|
|
furi_timer_stop(hid_mouse_clicker->timer);
|
|
}
|
|
|
|
with_view_model(
|
|
hid_mouse_clicker->view,
|
|
HidMouseClickerModel * model,
|
|
{
|
|
furi_timer_start(
|
|
hid_mouse_clicker->timer, furi_kernel_get_tick_frequency() / model->rate);
|
|
},
|
|
true);
|
|
}
|
|
|
|
static void hid_mouse_clicker_draw_callback(Canvas* canvas, void* context) {
|
|
furi_assert(context);
|
|
HidMouseClickerModel* model = context;
|
|
|
|
// Header
|
|
if(model->transport == HidTransportBle) {
|
|
if(model->connected) {
|
|
canvas_draw_icon(canvas, 0, 0, &I_Ble_connected_15x15);
|
|
} else {
|
|
canvas_draw_icon(canvas, 0, 0, &I_Ble_disconnected_15x15);
|
|
}
|
|
}
|
|
|
|
canvas_set_font(canvas, FontPrimary);
|
|
elements_multiline_text_aligned(canvas, 17, 3, AlignLeft, AlignTop, "Mouse Clicker");
|
|
|
|
// Ok
|
|
canvas_draw_icon(canvas, 63, 25, &I_Space_65x18);
|
|
if(model->running) {
|
|
canvas_set_font(canvas, FontPrimary);
|
|
|
|
FuriString* rate_label = furi_string_alloc();
|
|
furi_string_printf(rate_label, "%d clicks/s\n\nUp / Down", model->rate);
|
|
elements_multiline_text(canvas, AlignLeft, 35, furi_string_get_cstr(rate_label));
|
|
canvas_set_font(canvas, FontSecondary);
|
|
furi_string_free(rate_label);
|
|
|
|
elements_slightly_rounded_box(canvas, 66, 27, 60, 13);
|
|
canvas_set_color(canvas, ColorWhite);
|
|
} else {
|
|
canvas_set_font(canvas, FontPrimary);
|
|
elements_multiline_text(canvas, AlignLeft, 35, "Press Start\nto start\nclicking");
|
|
canvas_set_font(canvas, FontSecondary);
|
|
}
|
|
canvas_draw_icon(canvas, 74, 29, &I_Ok_btn_9x9);
|
|
if(model->running) {
|
|
elements_multiline_text_aligned(canvas, 91, 36, AlignLeft, AlignBottom, "Stop");
|
|
} else {
|
|
elements_multiline_text_aligned(canvas, 91, 36, AlignLeft, AlignBottom, "Start");
|
|
}
|
|
canvas_set_color(canvas, ColorBlack);
|
|
|
|
// Back
|
|
canvas_draw_icon(canvas, 74, 49, &I_Pin_back_arrow_10x8);
|
|
elements_multiline_text_aligned(canvas, 91, 57, AlignLeft, AlignBottom, "Quit");
|
|
}
|
|
|
|
static void hid_mouse_clicker_timer_callback(void* context) {
|
|
furi_assert(context);
|
|
HidMouseClicker* hid_mouse_clicker = context;
|
|
with_view_model(
|
|
hid_mouse_clicker->view,
|
|
HidMouseClickerModel * model,
|
|
{
|
|
if(model->running) {
|
|
hid_hal_mouse_press(hid_mouse_clicker->hid, HID_MOUSE_BTN_LEFT);
|
|
hid_hal_mouse_release(hid_mouse_clicker->hid, HID_MOUSE_BTN_LEFT);
|
|
}
|
|
},
|
|
false);
|
|
}
|
|
|
|
static void hid_mouse_clicker_enter_callback(void* context) {
|
|
hid_mouse_clicker_start_or_restart_timer(context);
|
|
}
|
|
|
|
static void hid_mouse_clicker_exit_callback(void* context) {
|
|
furi_assert(context);
|
|
HidMouseClicker* hid_mouse_clicker = context;
|
|
furi_timer_stop(hid_mouse_clicker->timer);
|
|
}
|
|
|
|
static bool hid_mouse_clicker_input_callback(InputEvent* event, void* context) {
|
|
furi_assert(context);
|
|
HidMouseClicker* hid_mouse_clicker = context;
|
|
|
|
bool consumed = false;
|
|
bool rate_changed = false;
|
|
|
|
if(event->type != InputTypeShort && event->type != InputTypeRepeat) {
|
|
return false;
|
|
}
|
|
|
|
with_view_model(
|
|
hid_mouse_clicker->view,
|
|
HidMouseClickerModel * model,
|
|
{
|
|
switch(event->key) {
|
|
case InputKeyOk:
|
|
model->running = !model->running;
|
|
consumed = true;
|
|
break;
|
|
case InputKeyUp:
|
|
if(model->rate < MAXIMUM_CLICK_RATE) {
|
|
model->rate++;
|
|
}
|
|
rate_changed = true;
|
|
consumed = true;
|
|
break;
|
|
case InputKeyDown:
|
|
if(model->rate > 1) {
|
|
model->rate--;
|
|
}
|
|
rate_changed = true;
|
|
consumed = true;
|
|
break;
|
|
default:
|
|
consumed = true;
|
|
break;
|
|
}
|
|
},
|
|
true);
|
|
|
|
if(rate_changed) {
|
|
hid_mouse_clicker_start_or_restart_timer(context);
|
|
}
|
|
|
|
return consumed;
|
|
}
|
|
|
|
HidMouseClicker* hid_mouse_clicker_alloc(Hid* hid) {
|
|
HidMouseClicker* hid_mouse_clicker = malloc(sizeof(HidMouseClicker));
|
|
|
|
hid_mouse_clicker->view = view_alloc();
|
|
view_set_context(hid_mouse_clicker->view, hid_mouse_clicker);
|
|
view_allocate_model(
|
|
hid_mouse_clicker->view, ViewModelTypeLocking, sizeof(HidMouseClickerModel));
|
|
view_set_draw_callback(hid_mouse_clicker->view, hid_mouse_clicker_draw_callback);
|
|
view_set_input_callback(hid_mouse_clicker->view, hid_mouse_clicker_input_callback);
|
|
view_set_enter_callback(hid_mouse_clicker->view, hid_mouse_clicker_enter_callback);
|
|
view_set_exit_callback(hid_mouse_clicker->view, hid_mouse_clicker_exit_callback);
|
|
|
|
hid_mouse_clicker->hid = hid;
|
|
|
|
hid_mouse_clicker->timer = furi_timer_alloc(
|
|
hid_mouse_clicker_timer_callback, FuriTimerTypePeriodic, hid_mouse_clicker);
|
|
|
|
with_view_model(
|
|
hid_mouse_clicker->view,
|
|
HidMouseClickerModel * model,
|
|
{
|
|
model->transport = hid->transport;
|
|
model->rate = DEFAULT_CLICK_RATE;
|
|
},
|
|
true);
|
|
|
|
return hid_mouse_clicker;
|
|
}
|
|
|
|
void hid_mouse_clicker_free(HidMouseClicker* hid_mouse_clicker) {
|
|
furi_assert(hid_mouse_clicker);
|
|
|
|
furi_timer_stop(hid_mouse_clicker->timer);
|
|
furi_timer_free(hid_mouse_clicker->timer);
|
|
|
|
view_free(hid_mouse_clicker->view);
|
|
|
|
free(hid_mouse_clicker);
|
|
}
|
|
|
|
View* hid_mouse_clicker_get_view(HidMouseClicker* hid_mouse_clicker) {
|
|
furi_assert(hid_mouse_clicker);
|
|
return hid_mouse_clicker->view;
|
|
}
|
|
|
|
void hid_mouse_clicker_set_connected_status(HidMouseClicker* hid_mouse_clicker, bool connected) {
|
|
furi_assert(hid_mouse_clicker);
|
|
with_view_model(
|
|
hid_mouse_clicker->view,
|
|
HidMouseClickerModel * model,
|
|
{ model->connected = connected; },
|
|
true);
|
|
}
|