#include "hid_media.h" #include #include #include #include #include "../hid.h" #include "hid_icons.h" #define TAG "HidMedia" struct HidMedia { View* view; Hid* hid; }; typedef struct { bool left_pressed; bool up_pressed; bool right_pressed; bool down_pressed; bool ok_pressed; bool connected; bool back_pressed; HidTransport transport; } HidMediaModel; static void hid_media_draw_arrow(Canvas* canvas, uint8_t x, uint8_t y, CanvasDirection dir) { canvas_draw_triangle(canvas, x, y, 5, 3, dir); if(dir == CanvasDirectionBottomToTop) { canvas_draw_dot(canvas, x, y - 1); } else if(dir == CanvasDirectionTopToBottom) { canvas_draw_dot(canvas, x, y + 1); } else if(dir == CanvasDirectionRightToLeft) { canvas_draw_dot(canvas, x - 1, y); } else if(dir == CanvasDirectionLeftToRight) { canvas_draw_dot(canvas, x + 1, y); } } static void hid_media_draw_callback(Canvas* canvas, void* context) { furi_assert(context); HidMediaModel* model = context; // Header if(model->transport == HidTransportBle) { if(model->connected) { canvas_draw_icon(canvas, 0, 0, &I_Ble_connected_15x15); } else { canvas_draw_icon(canvas, 0, 0, &I_Ble_disconnected_15x15); } } canvas_set_font(canvas, FontPrimary); elements_multiline_text_aligned(canvas, 17, 3, AlignLeft, AlignTop, "Media"); canvas_set_font(canvas, FontSecondary); // Keypad circles canvas_draw_icon(canvas, 58, 3, &I_OutCircles); // Up if(model->up_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 68, 6, &I_S_UP); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } canvas_draw_icon(canvas, 79, 9, &I_Volup_8x6); canvas_set_color(canvas, ColorBlack); // Down if(model->down_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 68, 36, &I_S_DOWN); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } canvas_draw_icon(canvas, 80, 41, &I_Voldwn_6x6); canvas_set_color(canvas, ColorBlack); // Left if(model->left_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 61, 13, &I_S_LEFT); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } hid_media_draw_arrow(canvas, 65, 28, CanvasDirectionRightToLeft); hid_media_draw_arrow(canvas, 70, 28, CanvasDirectionRightToLeft); canvas_set_color(canvas, ColorBlack); // Right if(model->right_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 91, 13, &I_S_RIGHT); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } hid_media_draw_arrow(canvas, 96, 28, CanvasDirectionLeftToRight); hid_media_draw_arrow(canvas, 101, 28, CanvasDirectionLeftToRight); canvas_set_color(canvas, ColorBlack); // Ok if(model->ok_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 74, 19, &I_Pressed_Button_19x19); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } hid_media_draw_arrow(canvas, 80, 28, CanvasDirectionLeftToRight); canvas_draw_line(canvas, 84, 26, 84, 30); canvas_draw_line(canvas, 86, 26, 86, 30); canvas_set_color(canvas, ColorBlack); // Exit if(model->back_pressed) { canvas_set_bitmap_mode(canvas, 1); canvas_draw_icon(canvas, 107, 33, &I_Pressed_Button_19x19); canvas_set_bitmap_mode(canvas, 0); canvas_set_color(canvas, ColorWhite); } canvas_draw_icon(canvas, 111, 38, &I_Pin_back_arrow_10x10); canvas_set_color(canvas, ColorBlack); canvas_draw_icon(canvas, 0, 54, &I_Pin_back_arrow_10x8); canvas_set_font(canvas, FontSecondary); elements_multiline_text_aligned(canvas, 13, 62, AlignLeft, AlignBottom, "Hold to exit"); } static void hid_media_process_press(HidMedia* hid_media, InputEvent* event) { with_view_model( hid_media->view, HidMediaModel * model, { if(event->key == InputKeyUp) { model->up_pressed = true; hid_hal_consumer_key_press(hid_media->hid, HID_CONSUMER_VOLUME_INCREMENT); } else if(event->key == InputKeyDown) { model->down_pressed = true; hid_hal_consumer_key_press(hid_media->hid, HID_CONSUMER_VOLUME_DECREMENT); } else if(event->key == InputKeyLeft) { model->left_pressed = true; hid_hal_consumer_key_press(hid_media->hid, HID_CONSUMER_SCAN_PREVIOUS_TRACK); } else if(event->key == InputKeyRight) { model->right_pressed = true; hid_hal_consumer_key_press(hid_media->hid, HID_CONSUMER_SCAN_NEXT_TRACK); } else if(event->key == InputKeyOk) { model->ok_pressed = true; hid_hal_consumer_key_press(hid_media->hid, HID_CONSUMER_PLAY_PAUSE); } else if(event->key == InputKeyBack) { model->back_pressed = true; } }, true); } static void hid_media_process_release(HidMedia* hid_media, InputEvent* event) { with_view_model( hid_media->view, HidMediaModel * model, { if(event->key == InputKeyUp) { model->up_pressed = false; hid_hal_consumer_key_release(hid_media->hid, HID_CONSUMER_VOLUME_INCREMENT); } else if(event->key == InputKeyDown) { model->down_pressed = false; hid_hal_consumer_key_release(hid_media->hid, HID_CONSUMER_VOLUME_DECREMENT); } else if(event->key == InputKeyLeft) { model->left_pressed = false; hid_hal_consumer_key_release(hid_media->hid, HID_CONSUMER_SCAN_PREVIOUS_TRACK); } else if(event->key == InputKeyRight) { model->right_pressed = false; hid_hal_consumer_key_release(hid_media->hid, HID_CONSUMER_SCAN_NEXT_TRACK); } else if(event->key == InputKeyOk) { model->ok_pressed = false; hid_hal_consumer_key_release(hid_media->hid, HID_CONSUMER_PLAY_PAUSE); } else if(event->key == InputKeyBack) { model->back_pressed = false; } }, true); } static bool hid_media_input_callback(InputEvent* event, void* context) { furi_assert(context); HidMedia* hid_media = context; bool consumed = false; if(event->type == InputTypePress) { hid_media_process_press(hid_media, event); consumed = true; } else if(event->type == InputTypeRelease) { hid_media_process_release(hid_media, event); consumed = true; } return consumed; } HidMedia* hid_media_alloc(Hid* hid) { HidMedia* hid_media = malloc(sizeof(HidMedia)); hid_media->view = view_alloc(); hid_media->hid = hid; view_set_context(hid_media->view, hid_media); view_allocate_model(hid_media->view, ViewModelTypeLocking, sizeof(HidMediaModel)); view_set_draw_callback(hid_media->view, hid_media_draw_callback); view_set_input_callback(hid_media->view, hid_media_input_callback); with_view_model( hid_media->view, HidMediaModel * model, { model->transport = hid->transport; }, true); return hid_media; } void hid_media_free(HidMedia* hid_media) { furi_assert(hid_media); view_free(hid_media->view); free(hid_media); } View* hid_media_get_view(HidMedia* hid_media) { furi_assert(hid_media); return hid_media->view; } void hid_media_set_connected_status(HidMedia* hid_media, bool connected) { furi_assert(hid_media); with_view_model( hid_media->view, HidMediaModel * model, { model->connected = connected; }, true); }