flashlight.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. // by @xMasterX
  2. #include <furi.h>
  3. #include <furi_hal_power.h>
  4. #include <gui/gui.h>
  5. #include <input/input.h>
  6. #include <stdlib.h>
  7. #include <gui/elements.h>
  8. #include "flashlight_icons.h"
  9. typedef enum {
  10. EventTypeTick,
  11. EventTypeKey,
  12. } EventType;
  13. typedef struct {
  14. EventType type;
  15. InputEvent input;
  16. } PluginEvent;
  17. typedef struct {
  18. FuriMutex* mutex;
  19. bool is_on;
  20. } PluginState;
  21. static void render_callback(Canvas* const canvas, void* ctx) {
  22. furi_assert(ctx);
  23. const PluginState* plugin_state = ctx;
  24. furi_mutex_acquire(plugin_state->mutex, FuriWaitForever);
  25. canvas_set_font(canvas, FontPrimary);
  26. elements_multiline_text_aligned(canvas, 64, 4, AlignCenter, AlignTop, "Flashlight");
  27. canvas_set_font(canvas, FontSecondary);
  28. canvas_draw_icon(canvas, 0, 17, &I_led_connections);
  29. if(!plugin_state->is_on) {
  30. elements_multiline_text_aligned(
  31. canvas, 64, 44, AlignCenter, AlignTop, "Press OK button turn on");
  32. } else {
  33. elements_multiline_text_aligned(canvas, 64, 38, AlignCenter, AlignTop, "Light is on!");
  34. elements_multiline_text_aligned(
  35. canvas, 64, 50, AlignCenter, AlignTop, "Press OK button to off");
  36. }
  37. furi_mutex_release(plugin_state->mutex);
  38. }
  39. static void input_callback(InputEvent* input_event, void* ctx) {
  40. furi_assert(ctx);
  41. FuriMessageQueue* event_queue = ctx;
  42. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  43. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  44. }
  45. static void flash_toggle(PluginState* const plugin_state) {
  46. furi_hal_gpio_write(&gpio_ext_pc3, false);
  47. furi_hal_gpio_init(&gpio_ext_pc3, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
  48. if(plugin_state->is_on) {
  49. furi_hal_gpio_write(&gpio_ext_pc3, false);
  50. plugin_state->is_on = false;
  51. } else {
  52. furi_hal_gpio_write(&gpio_ext_pc3, true);
  53. plugin_state->is_on = true;
  54. }
  55. }
  56. int32_t flashlight_app() {
  57. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  58. PluginState* plugin_state = malloc(sizeof(PluginState));
  59. plugin_state->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  60. if(!plugin_state->mutex) {
  61. FURI_LOG_E("flashlight", "cannot create mutex\r\n");
  62. furi_message_queue_free(event_queue);
  63. free(plugin_state);
  64. return 255;
  65. }
  66. // Set system callbacks
  67. ViewPort* view_port = view_port_alloc();
  68. view_port_draw_callback_set(view_port, render_callback, plugin_state);
  69. view_port_input_callback_set(view_port, input_callback, event_queue);
  70. // Open GUI and register view_port
  71. Gui* gui = furi_record_open(RECORD_GUI);
  72. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  73. PluginEvent event;
  74. for(bool processing = true; processing;) {
  75. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  76. furi_mutex_acquire(plugin_state->mutex, FuriWaitForever);
  77. if(event_status == FuriStatusOk) {
  78. // press events
  79. if(event.type == EventTypeKey) {
  80. if(event.input.type == InputTypePress) {
  81. switch(event.input.key) {
  82. case InputKeyUp:
  83. case InputKeyDown:
  84. case InputKeyRight:
  85. case InputKeyLeft:
  86. break;
  87. case InputKeyOk:
  88. flash_toggle(plugin_state);
  89. break;
  90. case InputKeyBack:
  91. processing = false;
  92. break;
  93. default:
  94. break;
  95. }
  96. }
  97. }
  98. }
  99. furi_mutex_release(plugin_state->mutex);
  100. view_port_update(view_port);
  101. }
  102. view_port_enabled_set(view_port, false);
  103. gui_remove_view_port(gui, view_port);
  104. furi_record_close(RECORD_GUI);
  105. view_port_free(view_port);
  106. furi_message_queue_free(event_queue);
  107. furi_mutex_free(plugin_state->mutex);
  108. return 0;
  109. }