caesar_cipher.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. #include <furi.h>
  2. #include <input/input.h>
  3. #include <stdlib.h>
  4. #include <gui/gui.h>
  5. #include <gui/view.h>
  6. #include <gui/view_dispatcher.h>
  7. #include <gui/modules/text_input.h>
  8. #include <gui/modules/text_box.h>
  9. #define TEXT_BUFFER_SIZE 256
  10. typedef enum {
  11. EventTypeTick,
  12. EventTypeKey,
  13. } EventType;
  14. typedef struct {
  15. EventType type;
  16. InputEvent input;
  17. } PluginEvent;
  18. typedef struct {
  19. FuriMutex* mutex;
  20. ViewDispatcher* view_dispatcher;
  21. TextInput* text_input;
  22. TextBox* text_box;
  23. char input[TEXT_BUFFER_SIZE];
  24. char output[(TEXT_BUFFER_SIZE * 26) + (26)]; // linebreaks
  25. } CaesarState;
  26. static void string_to_uppercase(char* input) {
  27. int i;
  28. for(i = 0; input[i] != '\0'; i++) {
  29. if(input[i] >= 'a' && input[i] <= 'z') {
  30. input[i] = input[i] - 32;
  31. } else {
  32. input[i] = input[i];
  33. }
  34. }
  35. }
  36. static void build_output(char* input, char* output) {
  37. int out = 0;
  38. for(int rot = 1; rot < 26; rot++) {
  39. int in;
  40. for(in = 0; input[in] != '\0'; in++) {
  41. if(input[in] >= 'A' && input[in] <= 'Z') {
  42. output[out] = 65 + (((input[in] - 65) + rot) % 26);
  43. } else {
  44. output[out] = input[in];
  45. }
  46. out++;
  47. }
  48. output[out] = '\n';
  49. out++;
  50. }
  51. output[out] = '\0';
  52. }
  53. static void text_input_callback(void* ctx) {
  54. furi_assert(ctx);
  55. CaesarState* caesar_state = ctx;
  56. furi_mutex_acquire(caesar_state->mutex, FuriWaitForever);
  57. FURI_LOG_D("caesar_cipher", "Input text: %s", caesar_state->input);
  58. // this is where we build the output.
  59. string_to_uppercase(caesar_state->input);
  60. FURI_LOG_D("caesar_cipher", "Upper text: %s", caesar_state->input);
  61. build_output(caesar_state->input, caesar_state->output);
  62. text_box_set_text(caesar_state->text_box, caesar_state->output);
  63. view_dispatcher_switch_to_view(caesar_state->view_dispatcher, 1);
  64. furi_mutex_release(caesar_state->mutex);
  65. }
  66. static bool back_event_callback(void* ctx) {
  67. const CaesarState* caesar_state = ctx;
  68. furi_mutex_acquire(caesar_state->mutex, FuriWaitForever);
  69. view_dispatcher_stop(caesar_state->view_dispatcher);
  70. furi_mutex_release(caesar_state->mutex);
  71. return true;
  72. }
  73. static void caesar_cipher_state_init(CaesarState* const caesar_state) {
  74. caesar_state->view_dispatcher = view_dispatcher_alloc();
  75. caesar_state->text_input = text_input_alloc();
  76. caesar_state->text_box = text_box_alloc();
  77. text_box_set_font(caesar_state->text_box, TextBoxFontText);
  78. }
  79. static void caesar_cipher_state_free(CaesarState* const caesar_state) {
  80. text_input_free(caesar_state->text_input);
  81. text_box_free(caesar_state->text_box);
  82. view_dispatcher_remove_view(caesar_state->view_dispatcher, 0);
  83. view_dispatcher_remove_view(caesar_state->view_dispatcher, 1);
  84. view_dispatcher_free(caesar_state->view_dispatcher);
  85. free(caesar_state);
  86. }
  87. int32_t caesar_cipher_app() {
  88. CaesarState* caesar_state = malloc(sizeof(CaesarState));
  89. FURI_LOG_D("caesar_cipher", "Running caesar_cipher_state_init");
  90. caesar_cipher_state_init(caesar_state);
  91. caesar_state->mutex = furi_mutex_alloc(FuriMutexTypeNormal);
  92. if(!caesar_state->mutex) {
  93. FURI_LOG_E("caesar_cipher", "cannot create mutex\r\n");
  94. free(caesar_state);
  95. return 255;
  96. }
  97. FURI_LOG_D("caesar_cipher", "Assigning text input callback");
  98. text_input_set_result_callback(
  99. caesar_state->text_input,
  100. text_input_callback,
  101. caesar_state,
  102. caesar_state->input,
  103. TEXT_BUFFER_SIZE,
  104. //clear default text
  105. true);
  106. text_input_set_header_text(caesar_state->text_input, "Input");
  107. // Open GUI and register view_port
  108. Gui* gui = furi_record_open("gui");
  109. //gui_add_view_port(gui, view_port, GuiLayerFullscreen);
  110. FURI_LOG_D("caesar_cipher", "Adding text input view to dispatcher");
  111. view_dispatcher_add_view(
  112. caesar_state->view_dispatcher, 0, text_input_get_view(caesar_state->text_input));
  113. view_dispatcher_add_view(
  114. caesar_state->view_dispatcher, 1, text_box_get_view(caesar_state->text_box));
  115. FURI_LOG_D("caesar_cipher", "Attaching view dispatcher to GUI");
  116. view_dispatcher_attach_to_gui(
  117. caesar_state->view_dispatcher, gui, ViewDispatcherTypeFullscreen);
  118. FURI_LOG_D("ceasar_cipher", "starting view dispatcher");
  119. view_dispatcher_set_navigation_event_callback(
  120. caesar_state->view_dispatcher, back_event_callback);
  121. view_dispatcher_set_event_callback_context(caesar_state->view_dispatcher, caesar_state);
  122. view_dispatcher_switch_to_view(caesar_state->view_dispatcher, 0);
  123. view_dispatcher_run(caesar_state->view_dispatcher);
  124. furi_record_close("gui");
  125. furi_mutex_free(caesar_state->mutex);
  126. caesar_cipher_state_free(caesar_state);
  127. return 0;
  128. }