#include #include "buttons.hpp" #include "globals.hpp" #include "hal/api.hpp" #include "app/app_manager.hpp" // From pico-watch.c: extern AppAPI app_api; //const uint BUTTON_PINS[] = {BUTTON_HOME, BUTTON_SELECT, BUTTON_MODE, BUTTON_UP, BUTTON_DOWN}; // Rising edge and falling edge are "inverted" as every button is set to be pulled up. This means that the falling edge is done when the button is pressed and the rising edge when the button is released. // SDL2 TODO: Implement button support #ifndef SDL2_BUILD void gpio_interrupt_cb(uint gpio, uint32_t events) { auto delta_since_press = time_since_button_press(); if (delta_since_press > g_s.button_delay_time) { if (app_api.m_interpret_button_press) { switch (events) { case GPIO_IRQ_EDGE_FALL: if (gpio == BUTTON_HOME && (g_s.foreground_app->app_get_attributes().id != 0)) // Home app app_mgr::app_switch_request(0); else app_mgr::app_btnpressed(g_s.foreground_app, gpio, delta_since_press); break; case GPIO_IRQ_EDGE_RISE: if (gpio != BUTTON_HOME) // For simplicity's sake app_mgr::app_btnreleased(g_s.foreground_app, gpio, delta_since_press); break; } } if (events == GPIO_IRQ_EDGE_FALL) app_api.button_last_set(gpio); g_s.button_last_pressed_time = to_ms_since_boot(get_absolute_time()); } } unsigned long time_since_button_press() { // FIXME: This does not correct overflows return to_ms_since_boot(get_absolute_time())-g_s.button_last_pressed_time; } void init_buttons() { gpio_init(BUTTON_HOME); gpio_pull_up(BUTTON_HOME); gpio_set_irq_enabled_with_callback(BUTTON_HOME, GPIO_IRQ_EDGE_FALL|GPIO_IRQ_EDGE_RISE, true, &gpio_interrupt_cb); gpio_init(BUTTON_SELECT); gpio_pull_up(BUTTON_SELECT); gpio_set_irq_enabled_with_callback(BUTTON_SELECT, GPIO_IRQ_EDGE_FALL|GPIO_IRQ_EDGE_RISE, true, &gpio_interrupt_cb); gpio_init(BUTTON_MODE); gpio_pull_up(BUTTON_MODE); gpio_set_irq_enabled_with_callback(BUTTON_MODE, GPIO_IRQ_EDGE_FALL|GPIO_IRQ_EDGE_RISE, true, &gpio_interrupt_cb); gpio_init(BUTTON_DOWN); gpio_pull_up(BUTTON_DOWN); gpio_set_irq_enabled_with_callback(BUTTON_DOWN, GPIO_IRQ_EDGE_FALL|GPIO_IRQ_EDGE_RISE, true, &gpio_interrupt_cb); gpio_init(BUTTON_UP); gpio_pull_up(BUTTON_UP); gpio_set_irq_enabled_with_callback(BUTTON_UP, GPIO_IRQ_EDGE_FALL|GPIO_IRQ_EDGE_RISE, true, &gpio_interrupt_cb); /* For some reason for loop does not work… TODO for (int i=0; i < NUMBER_OF_BUTTONS; i++) { uint button = BUTTON_PINS[i]; gpio_init(button); gpio_pull_up(button); gpio_set_irq_enabled_with_callback(button, GPIO_IRQ_EDGE_FALL , true, &gpio_interrupt_cb); } */ g_s.button_last_pressed_time = to_ms_since_boot(get_absolute_time()); } #else // SDL2_BUILD // this function will have to be called by SDL2 on keyboard/button press // gpio: button pressed // events: 0 is when button released and 1 is when button pressed void gpio_interrupt_cb(uint gpio, uint32_t events) { auto delta_since_press = time_since_button_press(); if (delta_since_press > g_s.button_delay_time) { if (app_api.m_interpret_button_press) { switch (events) { case 1: if (gpio == BUTTON_HOME && (g_s.foreground_app->app_get_attributes().id != 0)) // Home app app_mgr::app_switch_request(0); else app_mgr::app_btnpressed(g_s.foreground_app, gpio, delta_since_press); break; case 0: if (gpio != BUTTON_HOME) // For simplicity's sake app_mgr::app_btnreleased(g_s.foreground_app, gpio, delta_since_press); break; } } if (events == 1) app_api.button_last_set(gpio); g_s.button_last_pressed_time = to_ms_since_boot(get_absolute_time()); } } unsigned long time_since_button_press() { // FIXME: Thi