mirror of
https://github.com/UberGuidoZ/Flipper.git
synced 2025-01-03 20:30:19 +00:00
409 lines
14 KiB
C
409 lines
14 KiB
C
#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) {
|
|
if(furi_hal_speaker_is_mine() || furi_hal_speaker_acquire(1000)) {
|
|
furi_hal_speaker_start(
|
|
current_tuning_note_freq(tuning_fork_state), tuning_fork_state->volume);
|
|
}
|
|
}
|
|
|
|
static void stop() {
|
|
if(furi_hal_speaker_is_mine()) {
|
|
furi_hal_speaker_stop();
|
|
furi_hal_speaker_release();
|
|
}
|
|
}
|
|
|
|
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;
|
|
default:
|
|
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;
|
|
default:
|
|
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;
|
|
default:
|
|
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;
|
|
}
|