toaruos/userspace/gui/demo/make-it-snow.c

233 lines
5.7 KiB
C

/*
* This is a port of some code from a Wayland drag-drop demo.
*
* Copyright (C) 2010 Kristian Høgsberg
* Copyright (C) 2013-2014 Kevin Lange
*
* Permission to use, copy, modify, distribute, and sell this software and its
* documentation for any purpose is hereby granted without fee, provided that
* the above copyright notice appear in all copies and that both that copyright
* notice and this permission notice appear in supporting documentation, and
* that the name of the copyright holders not be used in advertising or
* publicity pertaining to distribution of the software without specific,
* written prior permission. The copyright holders make no representations
* about the suitability of this software for any purpose. It is provided "as
* is" without express or implied warranty.
*
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
* EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
* OF THIS SOFTWARE.
*/
#include <syscall.h>
#include <math.h>
#include <stdio.h>
#include <cairo.h>
#include "lib/yutani.h"
#include "lib/graphics.h"
#include "lib/pthread.h"
#include "lib/list.h"
static yutani_t * yctx;
static yutani_window_t * window;
static gfx_context_t * ctx;
static int should_exit = 0;
static list_t * snowflakes;
static int width, height;
struct snowflake {
int x;
int y;
cairo_surface_t * surface;
};
#define random rand
#define item_width 64
#define item_height 64
static int windspeed = 2;
static int gravity = 5;
static struct snowflake * create_snowflake() {
struct snowflake * item = malloc(sizeof(struct snowflake));
item->x = random() % width;
item->y = random() % height;
const int petal_count = 3 + random() % 5;
const double r1 = 20 + random() % 10;
const double r2 = 5 + random() % 12;
const double u = (10 + random() % 90) / 100.0;
const double v = (random() % 90) / 100.0;
cairo_t *cr;
int i;
double t, dt = 2 * M_PI / (petal_count * 2);
double x1, y1, x2, y2, x3, y3;
item->surface = cairo_image_surface_create(CAIRO_FORMAT_ARGB32, item_width, item_height);
cr = cairo_create(item->surface);
cairo_set_operator(cr, CAIRO_OPERATOR_SOURCE);
cairo_set_source_rgba(cr, 0, 0, 0, 0);
cairo_paint(cr);
cairo_set_operator(cr, CAIRO_OPERATOR_OVER);
cairo_translate(cr, item_width / 2, item_height / 2);
t = random();
cairo_move_to(cr, cos(t) * r1, sin(t) * r1);
for (i = 0; i < petal_count; i++, t += dt * 2) {
x1 = cos(t) * r1;
y1 = sin(t) * r1;
x2 = cos(t + dt) * r2;
y2 = sin(t + dt) * r2;
x3 = cos(t + 2 * dt) * r1;
y3 = sin(t + 2 * dt) * r1;
cairo_curve_to(cr,
x1 - y1 * u, y1 + x1 * u,
x2 + y2 * v, y2 - x2 * v,
x2, y2);
cairo_curve_to(cr,
x2 - y2 * v, y2 + x2 * v,
x3 + y3 * u, y3 - x3 * u,
x3, y3);
}
cairo_close_path(cr);
cairo_set_source_rgba(cr,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 100) / 99.0);
cairo_fill_preserve(cr);
cairo_set_line_width(cr, 1);
cairo_set_source_rgba(cr,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 50) / 49.0,
0.5 + (random() % 100) / 99.0);
cairo_stroke(cr);
cairo_destroy(cr);;
return item;
}
static void render() {
/* Clear window */
draw_fill(ctx, rgba(0,0,0,0));
/* Set up cairo context */
int stride = cairo_format_stride_for_width(CAIRO_FORMAT_ARGB32, window->width);
cairo_surface_t * surface = cairo_image_surface_create_for_data(ctx->backbuffer, CAIRO_FORMAT_ARGB32, window->width, window->height, stride);
cairo_t * cr = cairo_create(surface);
/* Draw some snowflakes */
foreach(node, snowflakes) {
struct snowflake * item = node->value;
cairo_save(cr);
cairo_set_source_surface(cr, item->surface, item->x, item->y);
cairo_paint(cr);
item->x += windspeed;
item->y += gravity;
if (item->y > height + item_height) {
item->y = -item_height;
}
if (item->x > width + item_width) {
item->x = -item_width;
}
cairo_restore(cr);
}
cairo_surface_flush(surface);
cairo_destroy(cr);
cairo_surface_flush(surface);
cairo_surface_destroy(surface);
flip(ctx);
}
void * draw_thread(void * garbage) {
(void)garbage;
while (!should_exit) {
render();
yutani_flip(yctx, window);
syscall_yield();
}
pthread_exit(0);
}
int main(int argc, char * argv[]) {
yctx = yutani_init();
width = yctx->display_width;
height = yctx->display_height;
window = yutani_window_create(yctx,width,height);
ctx = init_graphics_yutani_double_buffer(window);
draw_fill(ctx, rgba(0,0,0,0));
flip(ctx);
yutani_flip(yctx, window);
yutani_window_advertise_icon(yctx, window, "Cairo Snow Demo", "snow");
snowflakes = list_create();
for (int i = 0; i < 100; ++i) {
list_insert(snowflakes, create_snowflake());
}
pthread_t thread;
pthread_create(&thread, NULL, draw_thread, NULL);
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.action == KEY_ACTION_DOWN) {
switch (ke->event.keycode) {
case 'q':
should_exit = 1;
free(m);
goto finish;
}
}
}
break;
case YUTANI_MSG_SESSION_END:
should_exit = 1;
goto finish;
default:
break;
}
free(m);
}
}
finish:
yutani_close(yctx, window);
return 0;
}