#include "image_viewer.h" #include "cursor.h" #include "direction.h" #include "display_utils.h" #include "image.h" #include "input.h" #include "logger.h" #include "image_loader.h" #include "coords.h" #include "mzapo_led_strip.h" #include #include #define COMMANDS_NUM 30 #define CURSOR_SHOW_DURATION 3 // seconds #define MAX_ZOOM 10 #define MIN_ZOOM 0.02 void loader_callback(void *state, double p) { mzapo_ledstrip_t *ledstrip = (mzapo_ledstrip_t*)state; ledstrip_progress_bar_step(ledstrip, p * LED_STRIP_COUNT); } image_viewer_t image_viewer_create(char *filename, display_t *display, logger_t *logger, mzapo_ledstrip_t ledstrip) { image_viewer_t viewer = { .display = display, .image = image_create(filename), .cursor = cursor_create(), .logger = logger, .display_region = image_region_create(0, 0, DISPLAY_WIDTH, DISPLAY_HEIGHT), .image_region = image_region_create(0, 0, 0, 0), .ledstrip = ledstrip, }; viewer.error = image_loader_load(&viewer.image, loader_callback, &ledstrip); ledstrip_clear(&ledstrip); if (viewer.error == IMERR_SUCCESS) { viewer.image_region = image_region_create(0, 0, viewer.image.width, viewer.image.height); viewer.scale = image_get_initial_zoom(&viewer.image); cursor_center(&viewer.cursor, viewer.image_region); } return viewer; } void image_viewer_destroy(image_viewer_t *viewer) { image_destroy(&viewer->image); } void command_handler_move_cursor(void *data, direction_t direction, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; cursor_hide(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); int32_t x = viewer->scale.x, y = viewer->scale.y; bool move_cursor = false; switch (viewer->mode) { case MOD_CURSOR: if (!cursor_move(&viewer->cursor, image_get_zoom_region(&viewer->image, viewer->scale), direction, amount / viewer->scale.scale)) { direction_move_xy(direction, &x, &y, amount / viewer->scale.scale); move_cursor = true; } break; case MOD_IMAGE: logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Moving image"); direction_move_xy(direction, &x, &y, amount / viewer->scale.scale); move_cursor = true; break; default: break; } if (x < 0) { x = 0; } if (y < 0) { y = 0; } viewer->scale.x = x; viewer->scale.y = y; if (move_cursor) { cursor_move(&viewer->cursor, image_get_zoom_region(&viewer->image, viewer->scale), direction, amount / viewer->scale.scale); } image_viewer_display_image(viewer); cursor_show(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); display_render(viewer->display); ledstrip_turn_on(&viewer->ledstrip, ((double)viewer->cursor.x / viewer->image.width) * LED_STRIP_COUNT, 1); } void command_handler_move_left(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t*)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Moving cursor to left by %d", amount); command_handler_move_cursor(data, LEFT, amount); } void command_handler_move_right(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Moving cursor to right by %d", amount); command_handler_move_cursor(data, RIGHT, amount); } void command_handler_move_up(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Moving cursor up by %d", amount); command_handler_move_cursor(data, UP, amount); } void command_handler_move_down(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Moving cursor down by %d", amount); command_handler_move_cursor(data, DOWN, amount); } void command_handler_exit(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; viewer->running = false; } void zoom(image_viewer_t *viewer, int amount) { double zoom = viewer->scale.scale; double new_zoom = zoom * (1 + amount * 0.05); if (new_zoom > MAX_ZOOM) { logger_warn(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Cannot zoom more than %f", MAX_ZOOM); new_zoom = MAX_ZOOM; } if (new_zoom < MIN_ZOOM) { logger_warn(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Cannot zoom less than %f", MIN_ZOOM); new_zoom = MIN_ZOOM; } image_zoom_t tmp = { .scale = new_zoom, .x = viewer->scale.x, .y = viewer->scale.y, }; image_region_t current_region = image_get_zoom_region(&viewer->image, viewer->scale); image_region_t new_region = image_get_zoom_region(&viewer->image, tmp); uint16_t scaled_w = (uint16_t)(zoom * viewer->image.width), scaled_h = (uint16_t)(zoom * viewer->image.height); int32_t beg_x = ((double)DISPLAY_WIDTH - (double)(scaled_w)) / 2; int32_t beg_y = ((double)DISPLAY_HEIGHT - (double)(scaled_h)) / 2; if (beg_x < 0) { beg_x = 0; } if (beg_y < 0) { beg_y = 0; } if (scaled_w > DISPLAY_WIDTH) { scaled_w = DISPLAY_WIDTH; } if (scaled_h > DISPLAY_HEIGHT) { scaled_h = DISPLAY_HEIGHT; } int32_t diff_w = current_region.width - new_region.width; int32_t diff_h = current_region.height - new_region.height; double side_percentage_x = (double)(viewer->cursor.x - current_region.x) / current_region.width; double side_percentage_y = (double)(viewer->cursor.y - current_region.y) / current_region.height; tmp.x += side_percentage_x * diff_w; tmp.y += side_percentage_y * diff_h; viewer->scale = tmp; image_viewer_display_image(viewer); cursor_show(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); } // TODO: implement zoom functions void command_handler_zoom_in(void *data, int amount) { // hide cursor, zoom, show cursor image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Zooming in by %d", amount); zoom(viewer, amount); } void command_handler_zoom_out(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Zooming out by %d", amount); zoom(viewer, -amount); } void command_handler_zoom_reset(void *data, int amount) { image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Zoom reset by %d", amount); viewer->scale = image_get_initial_zoom(&viewer->image); image_viewer_display_image(viewer); cursor_center(&viewer->cursor, viewer->image_region); cursor_show(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); } void command_handler_change_mode(void *data, int amount) { if (!amount) { return; } image_viewer_t *viewer = (image_viewer_t *)data; logger_debug(viewer->logger, __FILE__, __FUNCTION__, __LINE__, "Changing mode"); uint8_t mode = viewer->mode; mode++; if (mode >= MOD_COUNT) { mode %= MOD_COUNT; } viewer->mode = mode; } void image_viewer_register_commands(image_viewer_t *viewer, commands_t *commands) { commands_register(commands, IN_KEYBOARD, 'h', &command_handler_move_left, viewer); commands_register(commands, IN_KEYBOARD, 'j', &command_handler_move_down, viewer); commands_register(commands, IN_KEYBOARD, 'k', &command_handler_move_up, viewer); commands_register(commands, IN_KEYBOARD, 'l', &command_handler_move_right, viewer); commands_register(commands, IN_KEYBOARD, 'e', &command_handler_exit, viewer); commands_register(commands, IN_KEYBOARD, 'z', &command_handler_zoom_in, viewer); commands_register(commands, IN_KEYBOARD, 'x', &command_handler_zoom_out, viewer); commands_register(commands, IN_KEYBOARD, 'r', &command_handler_zoom_reset, viewer); commands_register(commands, IN_KEYBOARD, 'm', &command_handler_change_mode, viewer); commands_register(commands, IN_ENCODER_ROTATE, 0, &command_handler_move_right, viewer); commands_register(commands, IN_ENCODER_ROTATE, 1, &command_handler_move_down, viewer); commands_register(commands, IN_ENCODER_ROTATE, 2, &command_handler_zoom_in, viewer); commands_register(commands, IN_ENCODER_CLICK, 0, &command_handler_exit, viewer); commands_register(commands, IN_ENCODER_CLICK, 1, &command_handler_change_mode, viewer); commands_register(commands, IN_ENCODER_CLICK, 2, &command_handler_zoom_reset, viewer); } void image_viewer_start_loop(image_viewer_t *viewer, void *reg_knobs_base) { command_t command_array[COMMANDS_NUM]; commands_t commands = commands_create(command_array, COMMANDS_NUM, reg_knobs_base); image_viewer_register_commands(viewer, &commands); viewer->running = true; time_t now = time(NULL); while (viewer->running) { // Main loop commands_check_input(&commands); now = time(NULL); if (viewer->cursor.shown && difftime(now, viewer->cursor.shown_at) >= CURSOR_SHOW_DURATION) { cursor_hide(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); display_render(viewer->display); } } } void image_viewer_display_image(image_viewer_t *viewer) { cursor_hide(&viewer->cursor, &viewer->image, viewer->scale, viewer->display); viewer->scale = image_write_to_display(&viewer->image, viewer->display, viewer->scale); display_render(viewer->display); }