flashlight.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  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, FuriMessageQueue* event_queue) {
  40. furi_assert(event_queue);
  41. PluginEvent event = {.type = EventTypeKey, .input = *input_event};
  42. furi_message_queue_put(event_queue, &event, FuriWaitForever);
  43. }
  44. static void flash_toggle(PluginState* const plugin_state) {
  45. furi_hal_gpio_write(&gpio_ext_pc3, false);
  46. furi_hal_gpio_init(&gpio_ext_pc3, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
  47. if(plugin_state->is_on) {
  48. furi_hal_gpio_write(&gpio_ext_pc3, false);
  49. plugin_state->is_on = false;
  50. } else {
  51. furi_hal_gpio_write(&gpio_ext_pc3, true);
  52. plugin_state->is_on = true;
  53. }
  54. }
  55. int32_t flashlight_app() {
  56. FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
  57. PluginState* plugin_state = malloc(sizeof(PluginState));
  58. plugin_state->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  59. if(!plugin_state->mutex) {
  60. FURI_LOG_E("flashlight", "cannot create mutex\r\n");
  61. furi_message_queue_free(event_queue);
  62. free(plugin_state);
  63. return 255;
  64. }
  65. // Set system callbacks
  66. ViewPort* view_port = view_port_alloc();
  67. view_port_draw_callback_set(view_port, render_callback, plugin_state);
  68. view_port_input_callback_set(view_port, input_callback, event_queue);
  69. // Open GUI and register view_port
  70. Gui* gui = furi_record_open(RECORD_GUI);
  71. gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  72. PluginEvent event;
  73. for(bool processing = true; processing;) {
  74. FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
  75. furi_mutex_acquire(plugin_state->mutex, FuriWaitForever);
  76. if(event_status == FuriStatusOk) {
  77. // press events
  78. if(event.type == EventTypeKey) {
  79. if(event.input.type == InputTypePress) {
  80. switch(event.input.key) {
  81. case InputKeyUp:
  82. case InputKeyDown:
  83. case InputKeyRight:
  84. case InputKeyLeft:
  85. break;
  86. case InputKeyOk:
  87. flash_toggle(plugin_state);
  88. break;
  89. case InputKeyBack:
  90. processing = false;
  91. break;
  92. default:
  93. break;
  94. }
  95. }
  96. }
  97. }
  98. furi_mutex_release(plugin_state->mutex);
  99. view_port_update(view_port);
  100. }
  101. view_port_enabled_set(view_port, false);
  102. gui_remove_view_port(gui, view_port);
  103. furi_record_close(RECORD_GUI);
  104. view_port_free(view_port);
  105. furi_message_queue_free(event_queue);
  106. furi_mutex_free(plugin_state->mutex);
  107. return 0;
  108. }