diff --git a/applications/applications.c b/applications/applications.c index 115eeadb4..8de2332d1 100644 --- a/applications/applications.c +++ b/applications/applications.c @@ -49,6 +49,7 @@ extern int32_t file_browser_app(void* p); // Plugins extern int32_t music_player_app(void* p); extern int32_t wav_player_app(void* p); +extern int32_t clock_app(void *p); // Games extern int32_t snake_game_app(void* p); @@ -210,6 +211,14 @@ const size_t FLIPPER_SYSTEM_APPS_COUNT = COUNT_OF(FLIPPER_SYSTEM_APPS); // Main menu APP const FlipperApplication FLIPPER_APPS[] = { +#ifdef APP_CLOCK + {.app = clock_app, + .name = "Clock", + .stack_size = 2048, + .icon = &A_Clock_14, + .flags = FlipperApplicationFlagDefault}, +#endif + #ifdef APP_SUBGHZ {.app = subghz_app, .name = "Sub-GHz", diff --git a/applications/applications.mk b/applications/applications.mk index b16bd1f65..35a1ff1cd 100644 --- a/applications/applications.mk +++ b/applications/applications.mk @@ -49,6 +49,7 @@ APP_MUSIC_PLAYER = 1 APP_SNAKE_GAME = 1 APP_WAV_PLAYER = 1 APP_TETRIS_GAME = 1 +APP_CLOCK = 1 # Debug APP_ACCESSOR = 1 @@ -254,6 +255,12 @@ CFLAGS += -DAPP_TETRIS_GAME SRV_GUI = 1 endif +APP_CLOCK ?= 0 +ifeq ($(APP_CLOCK), 1) +CFLAGS += -DAPP_CLOCK +SRV_GUI = 1 +endif + APP_IBUTTON ?= 0 ifeq ($(APP_IBUTTON), 1) CFLAGS += -DAPP_IBUTTON diff --git a/applications/clock_app/clock_app.c b/applications/clock_app/clock_app.c new file mode 100644 index 000000000..4adacb889 --- /dev/null +++ b/applications/clock_app/clock_app.c @@ -0,0 +1,282 @@ +#include +#include +#include +#include +#include +#include +#include + +#define TAG "Clock" + +bool timerStarted=false; +int timerSecs=0; +int songSelect=2; + +typedef enum { + EventTypeTick, + EventTypeKey, +} EventType; + +typedef struct { + EventType type; + InputEvent input; +} PluginEvent; + +typedef struct { + FuriHalRtcDateTime datetime; +} ClockState; + +static void clock_input_callback(InputEvent* input_event, osMessageQueueId_t event_queue) { + furi_assert(event_queue); + PluginEvent event = {.type = EventTypeKey, .input = *input_event}; + osMessageQueuePut(event_queue, &event, 0, osWaitForever); +} + +static void clock_render_callback(Canvas* const canvas, void* ctx) { + canvas_clear(canvas); + canvas_set_color(canvas, ColorBlack); + ClockState* state = (ClockState*)acquire_mutex((ValueMutex*)ctx, 25); + char strings[3][20]; + int curMin = (timerSecs/60); + int curSec = timerSecs-(curMin *60); + sprintf(strings[0], "%.4d-%.2d-%.2d", state->datetime.year, state->datetime.month, state->datetime.day); + sprintf(strings[1], "%.2d:%.2d:%.2d", state->datetime.hour, state->datetime.minute, state->datetime.second); + sprintf(strings[2], "%.2d:%.2d", curMin , curSec); + release_mutex((ValueMutex*)ctx, state); + canvas_set_font(canvas, FontBigNumbers); + canvas_draw_str_aligned(canvas, 64, 8, AlignCenter, AlignCenter, strings[1]); + canvas_set_font(canvas, FontSecondary); + canvas_draw_str_aligned(canvas, 64, 20, AlignCenter, AlignTop, strings[0]); + // elements_button_left(canvas, "Alarms"); + // elements_button_right(canvas, "Settings"); + // elements_button_center(canvas, "Reset"); + canvas_set_font(canvas, FontBigNumbers); + canvas_draw_str_aligned(canvas, 64, 32, AlignCenter, AlignTop, strings[2]); + canvas_set_font(canvas, FontSecondary); + if(timerStarted) { + elements_button_center(canvas, "Stop"); + } else { + elements_button_center(canvas, "Start"); + } + if(songSelect==0) { + elements_button_right(canvas, "S:OFF"); + } else if(songSelect==1) { + elements_button_right(canvas, "S:PoRa"); + } else if(songSelect==2) { + elements_button_right(canvas, "S:Mario"); + } else if(songSelect==3) { + elements_button_right(canvas, "S:ByMin"); + } +} + +static void clock_state_init(ClockState* const state) { + furi_hal_rtc_get_datetime(&state->datetime); +} + +const NotificationSequence clock_alert_silent = { + &message_force_vibro_setting_on, &message_vibro_on, &message_red_255, &message_green_255, &message_blue_255, &message_display_backlight_on, + &message_vibro_off, &message_display_backlight_off, &message_delay_50, &message_display_backlight_on, NULL, +}; +const NotificationSequence clock_alert_pr1 = { + &message_force_speaker_volume_setting_1f, + &message_force_vibro_setting_on, &message_vibro_on, &message_red_255, &message_green_255, &message_blue_255, &message_display_backlight_on, + &message_note_g5, &message_delay_100, &message_delay_100, &message_delay_50, &message_sound_off, + &message_vibro_off, &message_display_backlight_off, &message_delay_50, &message_display_backlight_on, + &message_note_g5, &message_delay_100, &message_delay_100, &message_delay_50, &message_sound_off, NULL, +}; +const NotificationSequence clock_alert_pr2 = { + &message_force_speaker_volume_setting_1f, + &message_force_vibro_setting_on, &message_vibro_on, + &message_note_fs5, &message_delay_100, &message_delay_100, &message_sound_off, + &message_display_backlight_off, &message_vibro_off, &message_delay_50, + &message_note_g5, &message_delay_100, &message_delay_100, &message_sound_off, + &message_display_backlight_on, &message_delay_50, + &message_note_a5, &message_delay_100, &message_delay_100, &message_sound_off, NULL, +}; +const NotificationSequence clock_alert_pr3 = { + &message_force_speaker_volume_setting_1f, + &message_display_backlight_off, + &message_note_g5, &message_delay_100, &message_delay_100, &message_sound_off, + &message_delay_50, &message_red_255, &message_green_255, &message_blue_255, &message_display_backlight_on, &message_delay_100, NULL, +}; +const NotificationSequence clock_alert_mario1 = { + &message_force_speaker_volume_setting_1f, + &message_force_vibro_setting_on, &message_vibro_on, &message_red_255, &message_green_255, &message_blue_255, &message_display_backlight_on, + &message_note_e5, &message_delay_100, &message_delay_100, &message_delay_50, &message_sound_off, + &message_note_e5, &message_delay_100, &message_delay_100, &message_delay_50, &message_sound_off, + &message_vibro_off, &message_display_backlight_off, &message_delay_100, &message_display_backlight_on, &message_delay_100, + &message_note_e5, &message_delay_100, &message_delay_100, &message_delay_50, &message_sound_off, NULL, +}; +const NotificationSequence clock_alert_mario2 = { + &message_force_speaker_volume_setting_1f, + &message_force_vibro_setting_on, &message_vibro_on, &message_display_backlight_off, &message_delay_100, &message_display_backlight_on, &message_delay_100, + &message_note_c5, &message_delay_100, &message_delay_100, &message_sound_off, + &message_display_backlight_off, &message_vibro_off, &message_delay_50, + &message_note_e5, &message_delay_100, &message_delay_100, &message_sound_off, + &message_display_backlight_on, NULL, +}; +const NotificationSequence clock_alert_mario3 = { + &message_force_speaker_volume_setting_1f, + &message_display_backlight_off, + &message_note_g5, &message_delay_100, &message_delay_100, &message_delay_100, &message_delay_100, &message_sound_off, + &message_delay_50, &message_red_255, &message_green_255, &message_blue_255, &message_display_backlight_on, &message_delay_100, + &message_note_g4, &message_delay_100, &message_delay_100, &message_delay_100, &message_delay_100, &message_sound_off, + NULL, +}; +const NotificationSequence clock_alert_perMin = { + &message_force_speaker_volume_setting_1f, + &message_note_g5, &message_delay_100, &message_delay_50, &message_sound_off, + &message_delay_10, + &message_note_g4, &message_delay_50, &message_delay_10, &message_delay_10, &message_sound_off, + NULL, +}; +const NotificationSequence clock_alert_startStop = { + &message_force_speaker_volume_setting_1f, + &message_note_d6, &message_delay_100, &message_delay_10, &message_delay_10, &message_sound_off, NULL, +}; + +// Runs every 1000ms by default +static void clock_tick(void* ctx) { + furi_assert(ctx); + osMessageQueueId_t event_queue = ctx; + PluginEvent event = {.type = EventTypeTick}; + if(timerStarted) { + timerSecs=timerSecs+1; + if(timerSecs%60==0 && songSelect!=0) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_perMin); + furi_record_close("notification"); + } + if(songSelect==1 ) { + if(timerSecs==80) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_pr1); + furi_record_close("notification"); + } + if(timerSecs==81) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_pr2); + furi_record_close("notification"); + } + if(timerSecs==82) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_pr3); + furi_record_close("notification"); + } + } else if(songSelect==2 ) { + if(timerSecs==80) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_mario1); + furi_record_close("notification"); + } + if(timerSecs==81) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_mario2); + furi_record_close("notification"); + } + if(timerSecs==82) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_mario3); + furi_record_close("notification"); + } + } else { + if(timerSecs==80) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_silent); + furi_record_close("notification"); + } + } + } + // It's OK to loose this event if system overloaded + osMessageQueuePut(event_queue, &event, 0, 0); +} + +int32_t clock_app(void* p) { + UNUSED(p); + timerStarted=false; + timerSecs=0; + songSelect=2; + osMessageQueueId_t event_queue = osMessageQueueNew(8, sizeof(PluginEvent), NULL); + ClockState* plugin_state = malloc(sizeof(ClockState)); + clock_state_init(plugin_state); + ValueMutex state_mutex; + if (!init_mutex(&state_mutex, plugin_state, sizeof(ClockState))) { + FURI_LOG_E(TAG, "cannot create mutex\r\n"); + free(plugin_state); + return 255; + } + // Set system callbacks + ViewPort* view_port = view_port_alloc(); + view_port_draw_callback_set(view_port, clock_render_callback, &state_mutex); + view_port_input_callback_set(view_port, clock_input_callback, event_queue); + osTimerId_t timer = osTimerNew(clock_tick, osTimerPeriodic, event_queue, NULL); + osTimerStart(timer, osKernelGetTickFreq()); + // Open GUI and register view_port + Gui* gui = furi_record_open("gui"); + gui_add_view_port(gui, view_port, GuiLayerFullscreen); + // Main loop + PluginEvent event; + for (bool processing = true; processing;) { + osStatus_t event_status = osMessageQueueGet(event_queue, &event, NULL, 100); + ClockState* plugin_state = (ClockState*)acquire_mutex_block(&state_mutex); + if (event_status == osOK) { + // press events + if (event.type == EventTypeKey) { + if (event.input.type == InputTypeShort || event.input.type == InputTypeRepeat) { + switch(event.input.key) { + case InputKeyUp: + if(timerStarted) timerSecs=timerSecs+5; + break; + case InputKeyDown: + if(timerStarted) timerSecs=timerSecs-5; + break; + case InputKeyRight: + if(songSelect==0) { + songSelect=1; + } else if(songSelect==1) { + songSelect=2; + } else if(songSelect==2) { + songSelect=3; + } else { + songSelect=0; + } + break; + case InputKeyLeft: + break; + case InputKeyOk: + if(songSelect==1 || songSelect==2 || songSelect==3) { + NotificationApp* notification = furi_record_open("notification"); + notification_message(notification, &clock_alert_startStop); + furi_record_close("notification"); + } + if(timerStarted) { + timerStarted=false; + timerSecs=0; + } else { + timerStarted=true; + } + break; + case InputKeyBack: + // Exit the plugin + processing = false; + break; + } + } + } else if(event.type == EventTypeTick) { + furi_hal_rtc_get_datetime(&plugin_state->datetime); + } + } else { + FURI_LOG_D(TAG, "osMessageQueue: event timeout"); + // event timeout + } + view_port_update(view_port); + release_mutex(&state_mutex, plugin_state); + } + osTimerDelete(timer); + view_port_enabled_set(view_port, false); + gui_remove_view_port(gui, view_port); + furi_record_close("gui"); + view_port_free(view_port); + osMessageQueueDelete(event_queue); + return 0; +} \ No newline at end of file diff --git a/assets/compiled/assets_icons.c b/assets/compiled/assets_icons.c index 0ac5b8257..f17cb85ed 100644 --- a/assets/compiled/assets_icons.c +++ b/assets/compiled/assets_icons.c @@ -325,6 +325,16 @@ const uint8_t _A_Bluetooth_14_4[] = {0x00,0x10,0x00,0x30,0x00,0x51,0x00,0x92,0x0 const uint8_t _A_Bluetooth_14_5[] = {0x00,0x10,0x00,0x30,0x00,0x51,0x00,0x92,0x00,0x94,0x04,0x58,0x08,0x30,0x09,0x30,0x09,0x58,0x08,0x94,0x04,0x92,0x00,0x51,0x00,0x30,0x00,0x10,0x00,}; const uint8_t* const _A_Bluetooth_14[] = {_A_Bluetooth_14_0,_A_Bluetooth_14_1,_A_Bluetooth_14_2,_A_Bluetooth_14_3,_A_Bluetooth_14_4,_A_Bluetooth_14_5}; +const uint8_t _A_Clock_14_0[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x49,0x12,0x41,0x10,0x41,0x10,0x01,0x10,0x09,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_1[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x13,0x81,0x10,0x41,0x10,0x01,0x10,0x09,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_2[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x12,0x01,0x10,0xC1,0x11,0x01,0x10,0x09,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_3[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x12,0x01,0x10,0x41,0x10,0x81,0x10,0x09,0x13,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_4[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x12,0x01,0x10,0x41,0x10,0x41,0x10,0x49,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_5[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x12,0x01,0x10,0x41,0x10,0x21,0x10,0x19,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_6[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x09,0x12,0x01,0x10,0x71,0x10,0x01,0x10,0x09,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t _A_Clock_14_7[] = {0x00,0xF0,0x01,0x0E,0x0E,0x47,0x1C,0x03,0x18,0x19,0x12,0x21,0x10,0x41,0x10,0x01,0x10,0x09,0x12,0x02,0x08,0x46,0x0C,0x0E,0x0E,0xF6,0x0D,0x02,0x08,}; +const uint8_t* const _A_Clock_14[] = {_A_Clock_14_0,_A_Clock_14_1,_A_Clock_14_2,_A_Clock_14_3,_A_Clock_14_4,_A_Clock_14_5,_A_Clock_14_6,_A_Clock_14_7}; + const uint8_t _A_Debug_14_0[] = {0x00,0x20,0x01,0xC1,0x20,0x22,0x11,0x24,0x09,0xD9,0x26,0x16,0x1A,0xD8,0x06,0xD8,0x06,0xD6,0x1A,0x19,0x26,0xE4,0x09,0xC2,0x10,0x01,0x20,0x00,0x00,}; const uint8_t _A_Debug_14_1[] = {0x00,0x20,0x01,0xC0,0x00,0x22,0x11,0x25,0x29,0xD8,0x06,0x16,0x1A,0xD9,0x26,0xD8,0x06,0xD4,0x0A,0x12,0x12,0xEA,0x15,0xC5,0x28,0x02,0x10,0x02,0x10,}; const uint8_t _A_Debug_14_2[] = {0x00,0x20,0x01,0xC0,0x00,0x20,0x01,0x24,0x09,0xDA,0x16,0x11,0x22,0xDC,0x0E,0xDA,0x16,0xD9,0x26,0x14,0x0A,0xF2,0x13,0xD1,0x22,0x08,0x04,0x06,0x18,}; @@ -780,6 +790,7 @@ const Icon I_KeySave_24x11 = {.width=24,.height=11,.frame_count=1,.frame_rate=0, const Icon A_125khz_14 = {.width=14,.height=14,.frame_count=4,.frame_rate=3,.frames=_A_125khz_14}; const Icon A_BadUsb_14 = {.width=14,.height=14,.frame_count=11,.frame_rate=3,.frames=_A_BadUsb_14}; const Icon A_Bluetooth_14 = {.width=14,.height=14,.frame_count=6,.frame_rate=3,.frames=_A_Bluetooth_14}; +const Icon A_Clock_14 = {.width=14,.height=14,.frame_count=8,.frame_rate=3,.frames=_A_Clock_14}; const Icon A_Debug_14 = {.width=14,.height=14,.frame_count=4,.frame_rate=3,.frames=_A_Debug_14}; const Icon A_FileManager_14 = {.width=14,.height=14,.frame_count=10,.frame_rate=3,.frames=_A_FileManager_14}; const Icon A_GPIO_14 = {.width=14,.height=14,.frame_count=8,.frame_rate=3,.frames=_A_GPIO_14}; diff --git a/assets/compiled/assets_icons.h b/assets/compiled/assets_icons.h index 26dbf94dd..e32ac4bb0 100644 --- a/assets/compiled/assets_icons.h +++ b/assets/compiled/assets_icons.h @@ -94,6 +94,7 @@ extern const Icon I_KeySave_24x11; extern const Icon A_125khz_14; extern const Icon A_BadUsb_14; extern const Icon A_Bluetooth_14; +extern const Icon A_Clock_14; extern const Icon A_Debug_14; extern const Icon A_FileManager_14; extern const Icon A_GPIO_14; diff --git a/assets/icons/MainMenu/Clock_14/frame_01.png b/assets/icons/MainMenu/Clock_14/frame_01.png new file mode 100644 index 000000000..80193281a Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_01.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_02.png b/assets/icons/MainMenu/Clock_14/frame_02.png new file mode 100644 index 000000000..7eb0f8bef Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_02.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_03.png b/assets/icons/MainMenu/Clock_14/frame_03.png new file mode 100644 index 000000000..5d4e6243b Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_03.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_04.png b/assets/icons/MainMenu/Clock_14/frame_04.png new file mode 100644 index 000000000..8578a5a73 Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_04.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_05.png b/assets/icons/MainMenu/Clock_14/frame_05.png new file mode 100644 index 000000000..9c50fe469 Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_05.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_06.png b/assets/icons/MainMenu/Clock_14/frame_06.png new file mode 100644 index 000000000..4fe0f2c63 Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_06.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_07.png b/assets/icons/MainMenu/Clock_14/frame_07.png new file mode 100644 index 000000000..8ff61ba2e Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_07.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_08.png b/assets/icons/MainMenu/Clock_14/frame_08.png new file mode 100644 index 000000000..baffdaa70 Binary files /dev/null and b/assets/icons/MainMenu/Clock_14/frame_08.png differ diff --git a/assets/icons/MainMenu/Clock_14/frame_rate b/assets/icons/MainMenu/Clock_14/frame_rate new file mode 100644 index 000000000..e440e5c84 --- /dev/null +++ b/assets/icons/MainMenu/Clock_14/frame_rate @@ -0,0 +1 @@ +3 \ No newline at end of file