2012-04-20 05:21:19 +04:00
|
|
|
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
|
|
|
*
|
|
|
|
* Wallpaper renderer.
|
|
|
|
*
|
|
|
|
*/
|
2012-03-08 08:31:24 +04:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <assert.h>
|
2012-09-20 08:16:21 +04:00
|
|
|
#include <unistd.h>
|
2012-03-08 08:31:24 +04:00
|
|
|
#include <math.h>
|
|
|
|
|
2014-04-16 06:45:56 +04:00
|
|
|
#include "lib/yutani.h"
|
2012-03-08 08:31:24 +04:00
|
|
|
#include "lib/graphics.h"
|
2012-10-16 06:23:47 +04:00
|
|
|
#include "lib/shmemfonts.h"
|
2012-03-08 08:31:24 +04:00
|
|
|
|
|
|
|
sprite_t * sprites[128];
|
|
|
|
sprite_t alpha_tmp;
|
|
|
|
|
2012-09-20 08:16:21 +04:00
|
|
|
#define ICON_X 24
|
|
|
|
#define ICON_TOP_Y 40
|
|
|
|
#define ICON_SPACING_Y 74
|
|
|
|
#define ICON_WIDTH 48
|
2014-04-20 04:41:29 +04:00
|
|
|
#define EXTRA_WIDTH 24
|
2012-09-20 08:16:21 +04:00
|
|
|
|
2012-03-08 08:31:24 +04:00
|
|
|
uint16_t win_width;
|
|
|
|
uint16_t win_height;
|
2014-04-20 04:41:29 +04:00
|
|
|
yutani_t * yctx;
|
|
|
|
yutani_window_t * wina;
|
Context-based graphics library.
All graphics library commands now take a gfx_context_t pointer, which
points to a simple datastructure describing a rendering context (width,
height, depth, total size, front buffer, backbuffer; where backbuffer =
front buffer when not in double-buffering mode, thus we always render to
backbuffer except on a flip). This may have caused a minor speed
reduction, but I don't really care as it's far more important that we
support multiple graphics contexts.
TODO:
- Shared Memory Fonts library (there are a couple of apps that use these
so-called "shmem fonts" on their own; we need a dedicated library for
them)
- Break off "TTK" GUI toolkit into its own library. Since it's just a
callback-based button framework, this shouldn't be too hard right now.
Also, with the previous tick, I'll be able to put labels on controls
and start using text in more places.
2012-04-17 22:21:34 +04:00
|
|
|
gfx_context_t * ctx;
|
2012-03-08 08:31:24 +04:00
|
|
|
|
|
|
|
int center_x(int x) {
|
|
|
|
return (win_width - x) / 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
int center_y(int y) {
|
|
|
|
return (win_height - y) / 2;
|
|
|
|
}
|
|
|
|
|
2014-04-20 04:41:29 +04:00
|
|
|
void init_sprite_png(int i, char * filename) {
|
|
|
|
sprites[i] = malloc(sizeof(sprite_t));
|
|
|
|
load_sprite_png(sprites[i], filename);
|
|
|
|
}
|
|
|
|
|
2012-09-20 08:16:21 +04:00
|
|
|
typedef struct {
|
|
|
|
char * icon;
|
|
|
|
char * appname;
|
|
|
|
char * title;
|
|
|
|
} application_t;
|
|
|
|
|
|
|
|
application_t applications[] = {
|
2014-05-04 00:20:16 +04:00
|
|
|
{"/usr/share/icons/48/utilities-terminal.png", "terminal", "Terminal"},
|
|
|
|
{"/usr/share/icons/48/applications-painting.png", "draw", "Draw!"},
|
|
|
|
{"/usr/share/icons/48/applications-simulation.png", "game", "RPG Demo"},
|
|
|
|
{"/usr/share/icons/48/julia.png", "julia", "Julia Fractals"},
|
2012-09-20 08:16:21 +04:00
|
|
|
{NULL, NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2012-09-13 11:29:29 +04:00
|
|
|
volatile int _continue = 1;
|
|
|
|
|
2012-09-20 08:16:21 +04:00
|
|
|
void launch_application(char * app) {
|
|
|
|
if (!fork()) {
|
|
|
|
char name[512];
|
|
|
|
sprintf(name, "/bin/%s", app);
|
|
|
|
printf("Starting %s\n", name);
|
|
|
|
char * args[] = {name, NULL};
|
2012-10-08 11:17:50 +04:00
|
|
|
execvp(args[0], args);
|
2012-09-20 08:16:21 +04:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
char * next_run_activate = NULL;
|
2014-04-20 04:41:29 +04:00
|
|
|
int focused_app = -1;
|
|
|
|
|
|
|
|
void redraw_apps(void) {
|
|
|
|
draw_sprite(ctx, sprites[1], 0, 0);
|
|
|
|
|
|
|
|
/* Load Application Shortcuts */
|
|
|
|
uint32_t i = 0;
|
|
|
|
while (1) {
|
|
|
|
if (!applications[i].icon) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
draw_sprite(ctx, sprites[i+2], ICON_X, ICON_TOP_Y + ICON_SPACING_Y * i);
|
|
|
|
|
|
|
|
uint32_t color = rgb(255,255,255);
|
|
|
|
|
|
|
|
if (i == focused_app) {
|
|
|
|
color = rgb(142,216,255);
|
|
|
|
}
|
|
|
|
|
|
|
|
int str_w = draw_string_width(applications[i].title) / 2;
|
|
|
|
int str_x = ICON_X + ICON_WIDTH / 2 - str_w;
|
|
|
|
int str_y = ICON_TOP_Y + ICON_SPACING_Y * i + ICON_WIDTH + 14;
|
|
|
|
draw_string_shadow(ctx, str_x, str_y, color, applications[i].title, rgb(0,0,0), 2, 1, 1, 3.0);
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
flip(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_focused(int i) {
|
|
|
|
if (focused_app != i) {
|
|
|
|
int old_focused = focused_app;
|
|
|
|
focused_app = i;
|
|
|
|
redraw_apps();
|
|
|
|
if (old_focused >= 0) {
|
|
|
|
yutani_flip_region(yctx, wina, 0, ICON_TOP_Y + ICON_SPACING_Y * old_focused, ICON_WIDTH + 2 * EXTRA_WIDTH, ICON_SPACING_Y);
|
|
|
|
}
|
|
|
|
if (focused_app >= 0) {
|
|
|
|
yutani_flip_region(yctx, wina, 0, ICON_TOP_Y + ICON_SPACING_Y * focused_app, ICON_WIDTH + 2 * EXTRA_WIDTH, ICON_SPACING_Y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-09-20 08:16:21 +04:00
|
|
|
|
2014-04-16 08:27:07 +04:00
|
|
|
void wallpaper_check_click(struct yutani_msg_window_mouse_event * evt) {
|
|
|
|
if (evt->command == YUTANI_MOUSE_EVENT_CLICK) {
|
2012-09-20 08:16:21 +04:00
|
|
|
printf("Click!\n");
|
|
|
|
if (evt->new_x > ICON_X && evt->new_x < ICON_X + ICON_WIDTH) {
|
|
|
|
uint32_t i = 0;
|
|
|
|
while (1) {
|
|
|
|
if (!applications[i].icon) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ((evt->new_y > ICON_TOP_Y + ICON_SPACING_Y * i) &&
|
|
|
|
(evt->new_y < ICON_TOP_Y + ICON_SPACING_Y + ICON_SPACING_Y * i)) {
|
|
|
|
printf("Launching application \"%s\"...\n", applications[i].title);
|
2014-04-16 08:27:07 +04:00
|
|
|
launch_application(applications[i].appname);
|
2012-09-20 08:16:21 +04:00
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
/* Within the icon range */
|
|
|
|
}
|
2014-04-21 00:44:17 +04:00
|
|
|
} else if (evt->command == YUTANI_MOUSE_EVENT_MOVE || evt->command == YUTANI_MOUSE_EVENT_ENTER) {
|
2014-04-20 04:41:29 +04:00
|
|
|
if (evt->new_x > 0 && evt->new_x < ICON_X + ICON_WIDTH + EXTRA_WIDTH) {
|
|
|
|
uint32_t i = 0;
|
|
|
|
while (1) {
|
|
|
|
if (!applications[i].icon) {
|
|
|
|
set_focused(-1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if ((evt->new_y > ICON_TOP_Y + ICON_SPACING_Y * i) &&
|
|
|
|
(evt->new_y < ICON_TOP_Y + ICON_SPACING_Y + ICON_SPACING_Y * i)) {
|
|
|
|
set_focused(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
/* Within the icon range */
|
|
|
|
} else {
|
|
|
|
set_focused(-1);
|
|
|
|
}
|
2014-04-21 00:44:17 +04:00
|
|
|
} else if (evt->command == YUTANI_MOUSE_EVENT_LEAVE) {
|
|
|
|
set_focused(-1);
|
2012-09-20 08:16:21 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-08 08:31:24 +04:00
|
|
|
int main (int argc, char ** argv) {
|
2014-04-20 04:41:29 +04:00
|
|
|
yctx = yutani_init();
|
2012-03-08 08:31:24 +04:00
|
|
|
|
2014-04-16 06:45:56 +04:00
|
|
|
int width = yctx->display_width;
|
|
|
|
int height = yctx->display_height;
|
2012-03-08 08:31:24 +04:00
|
|
|
|
2012-10-17 07:42:48 +04:00
|
|
|
char f_name[512];
|
|
|
|
sprintf(f_name, "%s/.wallpaper.png", getenv("HOME"));
|
|
|
|
FILE * f = fopen(f_name, "r");
|
|
|
|
if (f) {
|
|
|
|
fclose(f);
|
|
|
|
init_sprite_png(0, f_name);
|
|
|
|
} else {
|
|
|
|
init_sprite_png(0, "/usr/share/wallpaper.png");
|
|
|
|
}
|
2012-11-17 06:37:39 +04:00
|
|
|
|
2014-04-20 04:41:29 +04:00
|
|
|
uint32_t i = 0;
|
|
|
|
while (1) {
|
|
|
|
if (!applications[i].icon) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
printf("Loading png %s\n", applications[i].icon);
|
|
|
|
init_sprite_png(i+2, applications[i].icon);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
2012-11-17 06:37:39 +04:00
|
|
|
float x = (float)width / (float)sprites[0]->width;
|
|
|
|
float y = (float)height / (float)sprites[0]->height;
|
|
|
|
|
|
|
|
int nh = (int)(x * (float)sprites[0]->height);
|
|
|
|
int nw = (int)(y * (float)sprites[0]->width);;
|
|
|
|
|
2014-04-02 08:39:08 +04:00
|
|
|
sprites[1] = create_sprite(width, height, ALPHA_OPAQUE);
|
|
|
|
gfx_context_t * tmp = init_graphics_sprite(sprites[1]);
|
|
|
|
|
2012-11-17 06:37:39 +04:00
|
|
|
if (nw > width) {
|
2014-04-02 08:39:08 +04:00
|
|
|
draw_sprite_scaled(tmp, sprites[0], (width - nw) / 2, 0, nw, height);
|
2012-11-17 06:37:39 +04:00
|
|
|
} else {
|
2014-04-02 08:39:08 +04:00
|
|
|
draw_sprite_scaled(tmp, sprites[0], 0, (height - nh) / 2, width, nh);
|
2012-11-17 06:37:39 +04:00
|
|
|
}
|
2014-04-02 08:39:08 +04:00
|
|
|
|
|
|
|
free(tmp);
|
|
|
|
|
|
|
|
win_width = width;
|
|
|
|
win_height = height;
|
|
|
|
|
2014-04-20 04:41:29 +04:00
|
|
|
wina = yutani_window_create(yctx, width, height);
|
2014-04-02 08:39:08 +04:00
|
|
|
assert(wina);
|
2014-04-16 06:45:56 +04:00
|
|
|
yutani_set_stack(yctx, wina, YUTANI_ZORDER_BOTTOM);
|
|
|
|
ctx = init_graphics_yutani_double_buffer(wina);
|
2012-10-16 06:23:47 +04:00
|
|
|
init_shmemfonts();
|
|
|
|
|
2014-04-20 04:41:29 +04:00
|
|
|
redraw_apps();
|
2014-04-16 06:45:56 +04:00
|
|
|
yutani_flip(yctx, wina);
|
2012-03-08 08:31:24 +04:00
|
|
|
|
2014-04-19 06:23:45 +04:00
|
|
|
while (_continue) {
|
2014-04-16 06:45:56 +04:00
|
|
|
yutani_msg_t * m = yutani_poll(yctx);
|
|
|
|
if (m) {
|
2014-04-19 06:23:45 +04:00
|
|
|
switch (m->type) {
|
|
|
|
case YUTANI_MSG_WINDOW_MOUSE_EVENT:
|
|
|
|
wallpaper_check_click((struct yutani_msg_window_mouse_event *)m->data);
|
|
|
|
break;
|
|
|
|
case YUTANI_MSG_SESSION_END:
|
|
|
|
_continue = 0;
|
|
|
|
break;
|
2014-04-16 06:45:56 +04:00
|
|
|
}
|
|
|
|
free(m);
|
2012-09-20 08:16:21 +04:00
|
|
|
}
|
2012-03-08 09:44:02 +04:00
|
|
|
}
|
|
|
|
|
2014-04-16 06:45:56 +04:00
|
|
|
yutani_close(yctx, wina);
|
2012-03-08 08:31:24 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|