Project

General

Profile

Statistics
| Revision:

root / proj / sprite.c @ 167

History | View | Annotate | Download (2.23 KB)

1
#include <lcom/lcf.h>
2

    
3
#include "sprite.h"
4

    
5
#include "graph.h"
6
#include "utils.h"
7
#include <math.h>
8

    
9
struct sprite{
10
    int16_t x, y;
11
    uint16_t w, h;
12
    int16_t u0, v0;
13
    double theta;
14
    uint8_t *map;
15
};
16

    
17
sprite_t* (sprite_ctor)(const char **xpm, int u0, int v0){
18
    sprite_t *ret = (sprite_t*)malloc(sizeof(sprite_t));
19
    if(ret == NULL) return NULL;
20
    enum xpm_image_type type = XPM_8_8_8_8;
21
    xpm_image_t img;
22
    ret->map = xpm_load((xpm_map_t)xpm, type, &img);
23
    if(ret->map == NULL){
24
        free(ret);
25
        return NULL;
26
    }
27
    ret->x = 0;
28
    ret->y = 0;
29
    ret->w = img.width;
30
    ret->h = img.height;
31
    ret->u0 = u0;
32
    ret->v0 = u0;
33
    ret->theta = 0;
34
    return ret;
35
}
36
void (sprite_dtor)(sprite_t *p){
37
    if(p == NULL) return;
38
    if(p->map) free(p->map);
39
    free(p);
40
}
41

    
42
void (sprite_set_x)     (sprite_t *p, int16_t x){ p->x = x; }
43
void (sprite_set_y)     (sprite_t *p, int16_t y){ p->y = y; }
44
void (sprite_set_pos)   (sprite_t *p, int16_t x, int16_t y){ sprite_set_x(p, x); sprite_set_y(p, y); }
45
void (sprite_set_angle) (sprite_t *p, double angle){ p->theta = angle; }
46
void (sprite_set_center)(sprite_t *p, int16_t u0, int16_t v0){ p->u0 = u0; p->v0 = v0; }
47

    
48
int sprite_get_x(const sprite_t *p){ return p->x; }
49
int sprite_get_y(const sprite_t *p){ return p->y; }
50
int sprite_get_w(const sprite_t *p){ return p->w; }
51
int sprite_get_h(const sprite_t *p){ return p->h; }
52

    
53
void (sprite_src2pic)(const sprite_t *p, int16_t x, int16_t y, int16_t *u, int16_t *v){
54
    int16_t dx = x - p->x;
55
    int16_t dy = y - p->y;
56
    double s = sin(p->theta);
57
    double c = cos(p->theta);
58
    *u = dx*c - dy*s + p->u0;
59
    *v = dx*s + dy*c + p->v0;
60
}
61

    
62
void (sprite_draw)(const sprite_t *p){
63
    const int16_t diag = sqrt(p->w*p->w + p->h*p->h)+2;
64
    int16_t u, v;
65
    for(int16_t y = max(0,p->y-diag); y < min(p->y+diag,graph_get_YRes()); ++y){
66
        for(int16_t x = max(0,p->x-diag); x < min(p->x+diag,graph_get_XRes()); ++x){
67
            sprite_src2pic(p, x, y, &u, &v);
68
            if(0 <= u && u < p->w && 0 <= v && v < p->h){
69
                uint8_t *m = p->map + (v*p->w + u)*4;
70
                uint32_t color = SET_RGB(*(m+2), *(m+1), *(m));
71
                graph_set_pixel_alpha(p->x + x, p->y + y, color, *(m+3));
72
            }
73
        }
74
    }
75
}