#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 <time.h>
#include <stdlib.h>
#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);
}
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);
ledstrip_turn_on(
&viewer->ledstrip,
((double)viewer->cursor.x / viewer->image.width) * LED_STRIP_COUNT, 1);
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);
}