365 lines
12 KiB
C
365 lines
12 KiB
C
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
||
* This file is part of ToaruOS and is released under the terms
|
||
* of the NCSA / University of Illinois License - see LICENSE.md
|
||
* Copyright (C) 2015 Kevin Lange
|
||
*
|
||
* Live CD Welcome Program
|
||
*
|
||
*/
|
||
#include <stdlib.h>
|
||
#include <stdio.h>
|
||
#include <unistd.h>
|
||
#include <syscall.h>
|
||
#include <signal.h>
|
||
#include <errno.h>
|
||
#include <sys/wait.h>
|
||
|
||
#include "lib/yutani.h"
|
||
#include "lib/graphics.h"
|
||
#include "lib/decorations.h"
|
||
#include "gui/ttk/ttk.h"
|
||
|
||
#include "lib/trace.h"
|
||
#define TRACE_APP_NAME "live-wizard"
|
||
|
||
#define WIZARD_WIDTH 640
|
||
#define WIZARD_HEIGHT 480
|
||
|
||
static yutani_t * yctx;
|
||
|
||
static yutani_window_t * win_hints;
|
||
static gfx_context_t * ctx_hints;
|
||
|
||
static yutani_window_t * win_wizard;
|
||
static gfx_context_t * ctx_wizard;
|
||
|
||
static cairo_surface_t * surface_hints;
|
||
static cairo_surface_t * surface_wizard;
|
||
|
||
static cairo_t * cr_hints;
|
||
static cairo_t * cr_wizard;
|
||
|
||
static int should_exit = 0;
|
||
static int current_frame = 0;
|
||
|
||
static int center_x(int x) {
|
||
return ((int)yctx->display_width - x) / 2;
|
||
}
|
||
|
||
static int center_y(int y) {
|
||
return ((int)yctx->display_height - y) / 2;
|
||
}
|
||
|
||
static int center_win_x(int x) {
|
||
return (win_wizard->width - x) / 2;
|
||
}
|
||
|
||
static int button_width = 100;
|
||
static int button_height = 32;
|
||
static int button_focused = 0;
|
||
static void draw_next_button(int is_exit) {
|
||
if (button_focused == 1) {
|
||
/* hover */
|
||
_ttk_draw_button_hover(cr_wizard, center_win_x(button_width), 400, button_width, button_height, is_exit ? "Exit" : "Next");
|
||
} else if (button_focused == 2) {
|
||
/* Down */
|
||
_ttk_draw_button_select(cr_wizard, center_win_x(button_width), 400, button_width, button_height, is_exit ? "Exit" : "Next");
|
||
} else {
|
||
/* Something else? */
|
||
_ttk_draw_button(cr_wizard, center_win_x(button_width), 400, button_width, button_height, is_exit ? "Exit" : "Next");
|
||
}
|
||
}
|
||
|
||
static void draw_centered_label(int y, int size, char * label) {
|
||
set_font_face(FONT_SANS_SERIF);
|
||
set_font_size(size);
|
||
|
||
int x = center_win_x(draw_string_width(label));
|
||
draw_string(ctx_wizard, x, y, rgb(0,0,0), label);
|
||
}
|
||
|
||
static char * LOGO = "/usr/share/logo_login.png";
|
||
static void draw_logo(void) {
|
||
sprite_t logo;
|
||
load_sprite_png(&logo, LOGO);
|
||
draw_sprite(ctx_wizard, &logo, center_win_x(logo.width), 50);
|
||
}
|
||
|
||
static void draw_arrow(int x, int y, int angle) {
|
||
|
||
cairo_save(cr_hints);
|
||
|
||
cairo_surface_t * arrow = cairo_image_surface_create_from_png("/usr/share/wizard-arrow.png");
|
||
int w, h;
|
||
w = cairo_image_surface_get_width(arrow);
|
||
h = cairo_image_surface_get_height(arrow);
|
||
|
||
cairo_translate(cr_hints, x, y);
|
||
|
||
cairo_rotate(cr_hints, (double)angle * M_PI / 179.0);
|
||
cairo_translate(cr_hints, -w, -h/2);
|
||
|
||
cairo_set_source_surface(cr_hints, arrow, 0, 0);
|
||
cairo_paint(cr_hints);
|
||
cairo_surface_destroy(arrow);
|
||
cairo_restore(cr_hints);
|
||
|
||
}
|
||
|
||
static void redraw(void) {
|
||
draw_fill(ctx_hints, premultiply(rgba(0,0,0,100)));
|
||
draw_fill(ctx_wizard, rgb(TTK_BACKGROUND_DEFAULT));
|
||
|
||
/* Draw the current tutorial frame */
|
||
render_decorations(win_wizard, ctx_wizard, "Welcome to とあるOS");
|
||
switch (current_frame) {
|
||
case 0:
|
||
/* Labels for Welcome to とあるOS! */
|
||
draw_logo();
|
||
draw_centered_label(100+70, 20, "Welcome to とあるOS!");
|
||
draw_centered_label(100+88, 12, "This tutorial will guide you through the features");
|
||
draw_centered_label(100+102,12, "of the operating system, as well as give you a feel");
|
||
draw_centered_label(100+116,12, "for the UI and design principles.");
|
||
draw_centered_label(100+180,12, "When you're ready to continue, press \"Next\".");
|
||
draw_centered_label(120+200,12, "https://github.com/klange/toaruos - http://toaruos.org");
|
||
draw_centered_label(120+220,12, "とあるOS is free software, released under the terms");
|
||
draw_centered_label(120+234,12, "of the NCSA/University of Illinois license.");
|
||
draw_next_button(0);
|
||
break;
|
||
case 1:
|
||
draw_logo();
|
||
draw_arrow(center_x(WIZARD_WIDTH) + 620, center_y(WIZARD_HEIGHT) - 5, 90);
|
||
draw_centered_label(100+70,12,"If you wish to exit the tutorial at any time, you can");
|
||
draw_centered_label(100+84,12,"click the × in the upper right corner of this window.");
|
||
draw_next_button(0);
|
||
break;
|
||
case 2:
|
||
draw_logo();
|
||
draw_centered_label(100+70, 12,"As a reminder, とあるOS is a hobby project with few developers.");
|
||
draw_centered_label(100+84, 12,"As such, do not expect things to work perfectly, or in some cases,");
|
||
draw_centered_label(100+98, 12,"at all, as the kernel and drivers are very much \"work-in-progress\".");
|
||
draw_next_button(0);
|
||
break;
|
||
case 3:
|
||
draw_arrow(110, 120, -135);
|
||
|
||
cairo_save(cr_hints);
|
||
cairo_set_operator(cr_hints, CAIRO_OPERATOR_SOURCE);
|
||
cairo_set_source_rgba(cr_hints, 0.0, 0.0, 0.0, 0.0);
|
||
cairo_translate(cr_hints, 48.5, 72.5);
|
||
cairo_arc(cr_hints, 0, 0, 50, 0, 2 * M_PI);
|
||
cairo_fill(cr_hints);
|
||
cairo_restore(cr_hints);
|
||
|
||
draw_centered_label(100+10, 12,"とあるOS aims to provide a Unix-like environment.");
|
||
draw_centered_label(100+24, 12,"You can find familiar command-line tools by opening a terminal.");
|
||
draw_centered_label(100+38, 12,"Application shortcuts on the desktop are opened with a single click.");
|
||
draw_centered_label(100+52, 12,"You can also find more graphical applications in the Applications menu.");
|
||
draw_next_button(0);
|
||
break;
|
||
case 4:
|
||
draw_logo();
|
||
draw_centered_label(100+70,12,"That's it for now!");
|
||
draw_centered_label(100+88,12,"You've finished the tutorial.");
|
||
draw_centered_label(100+102,12,"More guides will be added to this tutorial in the future, but that's");
|
||
draw_centered_label(100+116,12,"all for now. Press 'Exit' to close the tutorial.");
|
||
draw_next_button(1);
|
||
break;
|
||
default:
|
||
exit(0);
|
||
break;
|
||
}
|
||
|
||
flip(ctx_hints);
|
||
flip(ctx_wizard);
|
||
yutani_flip(yctx, win_hints);
|
||
yutani_flip(yctx, win_wizard);
|
||
}
|
||
|
||
static void do_click_callback(void) {
|
||
current_frame += 1;
|
||
redraw();
|
||
}
|
||
|
||
static int previous_buttons = 0;
|
||
static void do_mouse_stuff(struct yutani_msg_window_mouse_event * me) {
|
||
if (button_focused == 2) {
|
||
/* See if we released and are still inside. */
|
||
if (me->command == YUTANI_MOUSE_EVENT_RAISE || me->command == YUTANI_MOUSE_EVENT_CLICK) {
|
||
if (!(me->buttons & YUTANI_MOUSE_BUTTON_LEFT)) {
|
||
if (me->new_x > center_win_x(button_width)
|
||
&& me->new_x < center_win_x(button_width) + button_width
|
||
&& me->new_y > 400
|
||
&& me->new_y < 400 + button_height) {
|
||
button_focused = 1;
|
||
do_click_callback();
|
||
} else {
|
||
button_focused = 0;
|
||
redraw();
|
||
}
|
||
}
|
||
}
|
||
} else {
|
||
if (me->new_x > center_win_x(button_width)
|
||
&& me->new_x < center_win_x(button_width) + button_width
|
||
&& me->new_y > 400
|
||
&& me->new_y < 400 + button_height) {
|
||
if (!button_focused) {
|
||
button_focused = 1;
|
||
redraw();
|
||
}
|
||
if (me->command == YUTANI_MOUSE_EVENT_DOWN && (me->buttons & YUTANI_MOUSE_BUTTON_LEFT)) {
|
||
button_focused = 2;
|
||
redraw();
|
||
}
|
||
} else {
|
||
if (button_focused) {
|
||
button_focused = 0;
|
||
redraw();
|
||
}
|
||
}
|
||
}
|
||
previous_buttons = me->buttons;
|
||
}
|
||
|
||
static void resize_finish(int xwidth, int xheight) {
|
||
yutani_window_resize_accept(yctx, win_hints, xwidth, xheight);
|
||
|
||
cairo_destroy(cr_hints);
|
||
cairo_surface_destroy(surface_hints);
|
||
|
||
reinit_graphics_yutani(ctx_hints, win_hints);
|
||
int stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, win_hints->width);
|
||
surface_hints = cairo_image_surface_create_for_data(ctx_hints->backbuffer, CAIRO_FORMAT_ARGB32, win_hints->width, win_hints->height, stride);
|
||
cr_hints = cairo_create(surface_hints);
|
||
|
||
yutani_window_resize_done(yctx, win_hints);
|
||
|
||
/* Re-center the wizard */
|
||
yutani_window_move(yctx, win_wizard, center_x(WIZARD_WIDTH), center_y(WIZARD_HEIGHT));
|
||
|
||
redraw();
|
||
}
|
||
|
||
int main(int argc, char * argv[]) {
|
||
|
||
TRACE("Opening some windows...");
|
||
yctx = yutani_init();
|
||
|
||
init_decorations();
|
||
|
||
win_hints = yutani_window_create_flags(yctx, yctx->display_width, yctx->display_height,
|
||
YUTANI_WINDOW_FLAG_NO_STEAL_FOCUS | YUTANI_WINDOW_FLAG_DISALLOW_DRAG |
|
||
YUTANI_WINDOW_FLAG_DISALLOW_RESIZE | YUTANI_WINDOW_FLAG_ALT_ANIMATION);
|
||
yutani_window_move(yctx, win_hints, 0, 0);
|
||
yutani_window_update_shape(yctx, win_hints, YUTANI_SHAPE_THRESHOLD_CLEAR);
|
||
ctx_hints = init_graphics_yutani_double_buffer(win_hints);
|
||
|
||
win_wizard = yutani_window_create_flags(yctx, WIZARD_WIDTH, WIZARD_HEIGHT,
|
||
YUTANI_WINDOW_FLAG_DISALLOW_DRAG | YUTANI_WINDOW_FLAG_DISALLOW_RESIZE);
|
||
yutani_window_move(yctx, win_wizard, center_x(WIZARD_WIDTH), center_y(WIZARD_HEIGHT));
|
||
ctx_wizard = init_graphics_yutani_double_buffer(win_wizard);
|
||
|
||
int stride;
|
||
|
||
stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, win_hints->width);
|
||
surface_hints = cairo_image_surface_create_for_data(ctx_hints->backbuffer, CAIRO_FORMAT_ARGB32, win_hints->width, win_hints->height, stride);
|
||
cr_hints = cairo_create(surface_hints);
|
||
|
||
stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, win_wizard->width);
|
||
surface_wizard = cairo_image_surface_create_for_data(ctx_wizard->backbuffer, CAIRO_FORMAT_ARGB32, win_wizard->width, win_wizard->height, stride);
|
||
cr_wizard = cairo_create(surface_wizard);
|
||
|
||
yutani_window_advertise_icon(yctx, win_wizard, "Welcome Tutorial", "live-welcome");
|
||
|
||
redraw();
|
||
|
||
yutani_focus_window(yctx, win_wizard->wid);
|
||
|
||
while (!should_exit) {
|
||
yutani_msg_t * m = yutani_poll(yctx);
|
||
|
||
if (m) {
|
||
switch (m->type) {
|
||
case YUTANI_MSG_KEY_EVENT:
|
||
{
|
||
struct yutani_msg_key_event * ke = (void*)m->data;
|
||
if (ke->event.key == 'q' && ke->event.action == KEY_ACTION_DOWN) {
|
||
should_exit = 1;
|
||
}
|
||
}
|
||
break;
|
||
case YUTANI_MSG_WINDOW_FOCUS_CHANGE:
|
||
{
|
||
struct yutani_msg_window_focus_change * wf = (void*)m->data;
|
||
if (wf->wid == win_hints->wid) {
|
||
yutani_focus_window(yctx, win_wizard->wid);
|
||
} else if (wf->wid == win_wizard->wid) {
|
||
win_wizard->focused = wf->focused;
|
||
redraw();
|
||
}
|
||
}
|
||
break;
|
||
case YUTANI_MSG_WINDOW_MOVE:
|
||
{
|
||
struct yutani_msg_window_move * wm = (void*)m->data;
|
||
if (wm->wid == win_hints->wid) {
|
||
if (wm->x != 0 || wm->y != 0) {
|
||
/* force us back to 0,0 */
|
||
yutani_window_move(yctx, win_hints, 0, 0);
|
||
}
|
||
} else if (wm->wid == win_wizard->wid) {
|
||
if (wm->x != center_x(WIZARD_WIDTH) || wm->y != center_y(WIZARD_HEIGHT)) {
|
||
yutani_window_move(yctx, win_wizard, center_x(WIZARD_WIDTH), center_y(WIZARD_HEIGHT));
|
||
}
|
||
}
|
||
}
|
||
break;
|
||
case YUTANI_MSG_WELCOME:
|
||
{
|
||
struct yutani_msg_welcome * mw = (void*)m->data;
|
||
|
||
fprintf(stderr, "ct display_width: %d\ndisplay_height: %d\n", yctx->display_width, yctx->display_height);
|
||
fprintf(stderr, "mw display_width: %d\ndisplay_height: %d\n", mw->display_width, mw->display_height);
|
||
yutani_window_resize(yctx, win_hints, mw->display_width, mw->display_height);
|
||
}
|
||
break;
|
||
case YUTANI_MSG_RESIZE_OFFER:
|
||
{
|
||
/* When we request a resize from the display-size-changed, we need
|
||
* to respond to the offer we'll get from the server to finish it */
|
||
struct yutani_msg_window_resize * wr = (void*)m->data;
|
||
if (wr->wid == win_hints->wid) {
|
||
resize_finish(wr->width, wr->height);
|
||
} /* Else, ignore resize offers for the main window */
|
||
}
|
||
break;
|
||
case YUTANI_MSG_WINDOW_MOUSE_EVENT:
|
||
{
|
||
struct yutani_msg_window_mouse_event * me = (void*)m->data;
|
||
if (me->wid != win_wizard->wid) break;
|
||
int result = decor_handle_event(yctx, m);
|
||
switch (result) {
|
||
case DECOR_CLOSE:
|
||
should_exit = 1;
|
||
break;
|
||
default:
|
||
/* Other actions */
|
||
do_mouse_stuff(me);
|
||
break;
|
||
}
|
||
}
|
||
break;
|
||
case YUTANI_MSG_SESSION_END:
|
||
should_exit = 1;
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
free(m);
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|