|
|
@@ -0,0 +1,395 @@
|
|
|
+#include <furi.h>
|
|
|
+#include <furi_hal.h>
|
|
|
+#include <input/input.h>
|
|
|
+#include <m-string.h>
|
|
|
+#include <string.h>
|
|
|
+#include <stdlib.h>
|
|
|
+
|
|
|
+#include <gui/gui.h>
|
|
|
+#include <gui/elements.h>
|
|
|
+#include <gui/canvas.h>
|
|
|
+
|
|
|
+#include <notification/notification.h>
|
|
|
+#include <notification/notification_messages.h>
|
|
|
+
|
|
|
+#include "notes.h"
|
|
|
+#include "tunings.h"
|
|
|
+
|
|
|
+typedef enum {
|
|
|
+ EventTypeTick,
|
|
|
+ EventTypeKey,
|
|
|
+} EventType;
|
|
|
+
|
|
|
+typedef struct {
|
|
|
+ EventType type;
|
|
|
+ InputEvent input;
|
|
|
+} PluginEvent;
|
|
|
+
|
|
|
+enum Page {
|
|
|
+ Tunings,
|
|
|
+ Notes
|
|
|
+};
|
|
|
+
|
|
|
+typedef struct {
|
|
|
+ bool playing;
|
|
|
+ enum Page page;
|
|
|
+ int current_tuning_note_index;
|
|
|
+ int current_tuning_index;
|
|
|
+ float volume;
|
|
|
+ TUNING tuning;
|
|
|
+} TuningForkState;
|
|
|
+
|
|
|
+static TUNING current_tuning(TuningForkState* tuningForkState) {
|
|
|
+ return tuningForkState->tuning;
|
|
|
+}
|
|
|
+
|
|
|
+static NOTE current_tuning_note(TuningForkState* tuningForkState) {
|
|
|
+ return current_tuning(tuningForkState).notes[tuningForkState->current_tuning_note_index];
|
|
|
+}
|
|
|
+
|
|
|
+static float current_tuning_note_freq(TuningForkState* tuningForkState) {
|
|
|
+ return current_tuning_note(tuningForkState).frequency;
|
|
|
+}
|
|
|
+
|
|
|
+static void current_tuning_note_label(TuningForkState* tuningForkState, char* outNoteLabel) {
|
|
|
+ for(int i=0; i < 20; ++i){
|
|
|
+ outNoteLabel[i] = current_tuning_note(tuningForkState).label[i];
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void current_tuning_label(TuningForkState* tuningForkState, char* outTuningLabel) {
|
|
|
+ for(int i=0; i < 20; ++i){
|
|
|
+ outTuningLabel[i] = current_tuning(tuningForkState).label[i];
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void updateTuning(TuningForkState* tuning_fork_state) {
|
|
|
+ tuning_fork_state->tuning = TuningList[tuning_fork_state->current_tuning_index];
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+}
|
|
|
+
|
|
|
+static void next_tuning(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->current_tuning_index == TUNINGS_COUNT - 1) {
|
|
|
+ tuning_fork_state->current_tuning_index = 0;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->current_tuning_index += 1;
|
|
|
+ }
|
|
|
+ updateTuning(tuning_fork_state);
|
|
|
+}
|
|
|
+
|
|
|
+static void prev_tuning(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->current_tuning_index - 1 < 0) {
|
|
|
+ tuning_fork_state->current_tuning_index = TUNINGS_COUNT - 1;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->current_tuning_index -= 1;
|
|
|
+ }
|
|
|
+ updateTuning(tuning_fork_state);
|
|
|
+}
|
|
|
+
|
|
|
+static void next_note(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->current_tuning_note_index == current_tuning(tuning_fork_state).notes_length - 1) {
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->current_tuning_note_index += 1;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void prev_note(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->current_tuning_note_index == 0) {
|
|
|
+ tuning_fork_state->current_tuning_note_index = current_tuning(tuning_fork_state).notes_length - 1;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->current_tuning_note_index -= 1;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void increase_volume(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->volume < 1.0f) {
|
|
|
+ tuning_fork_state->volume += 0.1f;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void decrease_volume(TuningForkState* tuning_fork_state) {
|
|
|
+ if (tuning_fork_state->volume > 0.0f) {
|
|
|
+ tuning_fork_state->volume -= 0.1f;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void play(TuningForkState* tuning_fork_state) {
|
|
|
+ furi_hal_speaker_start(current_tuning_note_freq(tuning_fork_state), tuning_fork_state->volume);
|
|
|
+}
|
|
|
+
|
|
|
+static void stop() {
|
|
|
+ furi_hal_speaker_stop();
|
|
|
+}
|
|
|
+
|
|
|
+static void replay(TuningForkState* tuning_fork_state) {
|
|
|
+ stop();
|
|
|
+ play(tuning_fork_state);
|
|
|
+}
|
|
|
+
|
|
|
+static void render_callback(Canvas* const canvas, void* ctx) {
|
|
|
+ TuningForkState* tuning_fork_state = acquire_mutex((ValueMutex*)ctx, 25);
|
|
|
+ if(tuning_fork_state == NULL) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ string_t tempStr;
|
|
|
+ string_init(tempStr);
|
|
|
+
|
|
|
+ canvas_draw_frame(canvas, 0, 0, 128, 64);
|
|
|
+
|
|
|
+ canvas_set_font(canvas, FontPrimary);
|
|
|
+
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ char tuningLabel[20];
|
|
|
+ current_tuning_label(tuning_fork_state, tuningLabel);
|
|
|
+ string_printf(tempStr, "< %s >", tuningLabel);
|
|
|
+ canvas_draw_str_aligned(canvas, 64, 28, AlignCenter, AlignCenter, string_get_cstr(tempStr));
|
|
|
+ string_reset(tempStr);
|
|
|
+ } else {
|
|
|
+ char tuningLabel[20];
|
|
|
+ current_tuning_label(tuning_fork_state, tuningLabel);
|
|
|
+ string_printf(tempStr, "%s", tuningLabel);
|
|
|
+ canvas_draw_str_aligned(canvas, 64, 8, AlignCenter, AlignCenter, string_get_cstr(tempStr));
|
|
|
+ string_reset(tempStr);
|
|
|
+
|
|
|
+ char tuningNoteLabel[20];
|
|
|
+ current_tuning_note_label(tuning_fork_state, tuningNoteLabel);
|
|
|
+ string_printf(tempStr, "< %s >", tuningNoteLabel);
|
|
|
+ canvas_draw_str_aligned(canvas, 64, 24, AlignCenter, AlignCenter, string_get_cstr(tempStr));
|
|
|
+ string_reset(tempStr);
|
|
|
+ }
|
|
|
+
|
|
|
+ canvas_set_font(canvas, FontSecondary);
|
|
|
+ elements_button_left(canvas, "Prev");
|
|
|
+ elements_button_right(canvas, "Next");
|
|
|
+
|
|
|
+ if (tuning_fork_state->page == Notes) {
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ elements_button_center(canvas, "Stop ");
|
|
|
+ } else {
|
|
|
+ elements_button_center(canvas, "Play");
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ elements_button_center(canvas, "Select");
|
|
|
+ }
|
|
|
+ if (tuning_fork_state->page == Notes) {
|
|
|
+ elements_progress_bar(canvas, 8, 36, 112, tuning_fork_state->volume);
|
|
|
+ }
|
|
|
+
|
|
|
+ string_clear(tempStr);
|
|
|
+ release_mutex((ValueMutex*)ctx, tuning_fork_state);
|
|
|
+}
|
|
|
+
|
|
|
+static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
|
|
|
+ furi_assert(event_queue);
|
|
|
+
|
|
|
+ PluginEvent event = {.type = EventTypeKey, .input = *input_event};
|
|
|
+ furi_message_queue_put(event_queue, &event, FuriWaitForever);
|
|
|
+}
|
|
|
+
|
|
|
+static void tuning_fork_state_init(TuningForkState* const tuning_fork_state) {
|
|
|
+ tuning_fork_state->playing = false;
|
|
|
+ tuning_fork_state->page = Tunings;
|
|
|
+ tuning_fork_state->volume = 1.0f;
|
|
|
+ tuning_fork_state->tuning = GuitarStandard6;
|
|
|
+ tuning_fork_state->current_tuning_index = 2;
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+}
|
|
|
+
|
|
|
+int32_t tuning_fork_app() {
|
|
|
+ FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
|
|
|
+
|
|
|
+ TuningForkState* tuning_fork_state = malloc(sizeof(TuningForkState));
|
|
|
+ tuning_fork_state_init(tuning_fork_state);
|
|
|
+
|
|
|
+ ValueMutex state_mutex;
|
|
|
+ if(!init_mutex(&state_mutex, tuning_fork_state, sizeof(TuningForkState))) {
|
|
|
+ FURI_LOG_E("TuningFork", "cannot create mutex\r\n");
|
|
|
+ free(tuning_fork_state);
|
|
|
+ return 255;
|
|
|
+ }
|
|
|
+
|
|
|
+ // Set system callbacks
|
|
|
+ ViewPort* view_port = view_port_alloc();
|
|
|
+ view_port_draw_callback_set(view_port, render_callback, &state_mutex);
|
|
|
+ view_port_input_callback_set(view_port, input_callback, event_queue);
|
|
|
+
|
|
|
+ Gui* gui = furi_record_open("gui");
|
|
|
+ gui_add_view_port(gui, view_port, GuiLayerFullscreen);
|
|
|
+
|
|
|
+ PluginEvent event;
|
|
|
+ for(bool processing = true; processing;) {
|
|
|
+ FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
|
|
|
+
|
|
|
+ TuningForkState* tuning_fork_state = (TuningForkState*)acquire_mutex_block(&state_mutex);
|
|
|
+
|
|
|
+ if(event_status == FuriStatusOk) {
|
|
|
+ if(event.type == EventTypeKey) {
|
|
|
+ if(event.input.type == InputTypeShort) {
|
|
|
+ // push events
|
|
|
+ switch(event.input.key) {
|
|
|
+ case InputKeyUp:
|
|
|
+ if (tuning_fork_state->page == Notes) {
|
|
|
+ increase_volume(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case InputKeyDown:
|
|
|
+ if (tuning_fork_state->page == Notes) {
|
|
|
+ decrease_volume(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case InputKeyRight:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ next_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ next_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case InputKeyLeft:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ prev_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ prev_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case InputKeyOk:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ tuning_fork_state->page = Notes;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->playing = !tuning_fork_state->playing;
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ play(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ stop();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case InputKeyBack:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ processing = false;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->playing = false;
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+ stop();
|
|
|
+ tuning_fork_state->page = Tunings;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ } else if (event.input.type == InputTypeLong) {
|
|
|
+ // hold events
|
|
|
+ switch(event.input.key) {
|
|
|
+ case InputKeyUp:
|
|
|
+ break;
|
|
|
+ case InputKeyDown:
|
|
|
+ break;
|
|
|
+ case InputKeyRight:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ next_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ next_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ break;
|
|
|
+ case InputKeyLeft:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ prev_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ prev_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ break;
|
|
|
+ case InputKeyOk:
|
|
|
+ break;
|
|
|
+ case InputKeyBack:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ processing = false;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->playing = false;
|
|
|
+ stop();
|
|
|
+ tuning_fork_state->page = Tunings;
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ } else if (event.input.type == InputTypeRepeat) {
|
|
|
+ // repeat events
|
|
|
+ switch(event.input.key) {
|
|
|
+ case InputKeyUp:
|
|
|
+ break;
|
|
|
+ case InputKeyDown:
|
|
|
+ break;
|
|
|
+ case InputKeyRight:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ next_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ next_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ break;
|
|
|
+ case InputKeyLeft:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ prev_tuning(tuning_fork_state);
|
|
|
+ } else {
|
|
|
+ prev_note(tuning_fork_state);
|
|
|
+ if (tuning_fork_state->playing) {
|
|
|
+ replay(tuning_fork_state);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ break;
|
|
|
+ case InputKeyOk:
|
|
|
+ break;
|
|
|
+ case InputKeyBack:
|
|
|
+ if (tuning_fork_state->page == Tunings) {
|
|
|
+ processing = false;
|
|
|
+ } else {
|
|
|
+ tuning_fork_state->playing = false;
|
|
|
+ stop();
|
|
|
+ tuning_fork_state->page = Tunings;
|
|
|
+ tuning_fork_state->current_tuning_note_index = 0;
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ FURI_LOG_D("TuningFork", "FuriMessageQueue: event timeout");
|
|
|
+ }
|
|
|
+
|
|
|
+ view_port_update(view_port);
|
|
|
+ release_mutex(&state_mutex, tuning_fork_state);
|
|
|
+ }
|
|
|
+
|
|
|
+ view_port_enabled_set(view_port, false);
|
|
|
+ gui_remove_view_port(gui, view_port);
|
|
|
+ furi_record_close("gui");
|
|
|
+ view_port_free(view_port);
|
|
|
+ furi_message_queue_free(event_queue);
|
|
|
+ delete_mutex(&state_mutex);
|
|
|
+ furi_record_close(RECORD_NOTIFICATION);
|
|
|
+ free(tuning_fork_state);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|