summaryrefslogtreecommitdiff
path: root/code/fe310/eos/eve_text.c
blob: 4e343a0340939dd8ceed988363d8e6a9ac44534a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
#include "eve.h"
#include "eve_text.h"

static int _line_diff(EVEText *box, int line) {
    if (line > box->line_idx) {
        return line - box->line_idx;
    } else {
        return box->buf_line_h - box->line_idx + line;
    }
}

static void _line_scroll(EVEText *box) {
    box->line_idx = (box->line_idx + 1) % box->buf_line_h;
    eos_eve_cmd(CMD_MEMSET, "www", box->buf_addr + box->buf_idx, 0x0, box->w * 2);
    eos_eve_cmd_exec(1);
    eos_eve_text_update(box);
}

void eos_eve_text_init(EVEText *box, int x, int y, int w, int h, uint8_t bitmap_handle, uint32_t mem_addr, int buf_line_h, uint32_t *mem_next) {
    box->x = x;
    box->y = y;
    box->w = w;
    box->h = h;
    box->bitmap_handle = bitmap_handle;
    box->buf_addr = mem_addr;
    box->buf_line_h = buf_line_h;
    box->buf_idx = 0;
    box->line_idx = 0;

    eos_eve_cmd(CMD_MEMSET, "www", mem_addr, 0x0, w * 2 * buf_line_h);
    eos_eve_cmd_exec(1);

    eos_eve_text_update(box);
    *mem_next = box->buf_addr + box->w * 2 * box->buf_line_h + 12 * 4;
}

void eos_eve_text_draw(EVEText *box) {
    eos_eve_cmd(CMD_APPEND, "ww", box->buf_addr + box->w * 2 * box->buf_line_h, 12 * 4);
}

void eos_eve_text_update(EVEText *box) {
    int text_h1;
    int text_h2;

    if (box->line_idx + box->h > box->buf_line_h) {
        text_h1 = box->buf_line_h - box->line_idx;
        text_h2 = box->h - text_h1;
    } else {
        text_h1 = box->h;
        text_h2 = 0;
    }
    eos_eve_dl_start(box->buf_addr + box->w * 2 * box->buf_line_h);
    eos_eve_dl_write(BITMAP_HANDLE(box->bitmap_handle));
    eos_eve_dl_write(BITMAP_SOURCE(box->buf_addr + box->line_idx * box->w * 2));
    eos_eve_dl_write(BITMAP_LAYOUT(EVE_TEXTVGA, box->w * 2, text_h1));
    eos_eve_dl_write(BITMAP_SIZE(EVE_NEAREST, EVE_BORDER, EVE_BORDER, box->w * 8, text_h1 * 16));

    if (text_h2) {
        eos_eve_dl_write(BITMAP_HANDLE(box->bitmap_handle+1));
        eos_eve_dl_write(BITMAP_SOURCE(box->buf_addr));
        eos_eve_dl_write(BITMAP_LAYOUT(EVE_TEXTVGA, box->w * 2, text_h2));
        eos_eve_dl_write(BITMAP_SIZE(EVE_NEAREST, EVE_BORDER, EVE_BORDER, box->w * 8, text_h2 * 16));
    } else {
        eos_eve_dl_write(NOP());
        eos_eve_dl_write(NOP());
        eos_eve_dl_write(NOP());
        eos_eve_dl_write(NOP());
    }

    eos_eve_dl_write(BEGIN(EVE_BITMAPS));
    eos_eve_dl_write(VERTEX2II(box->x, box->y, 0, 0));
    if (text_h2) {
        eos_eve_dl_write(VERTEX2II(box->x, box->y + text_h1 * 16, 1, 0));
    } else {
        eos_eve_dl_write(NOP());
    }
    eos_eve_dl_write(END());
}

void eos_eve_text_putc(EVEText *box, int c) {
    int line_c, line_n;

    line_c = box->buf_idx / 2 / box->w;

    eos_eve_write16(box->buf_addr + box->buf_idx, 0x0A00 | (c & 0xff));
    box->buf_idx = (box->buf_idx + 2) % (box->buf_line_h * box->w * 2);

    line_n = box->buf_idx / 2 / box->w;
    if ((line_c != line_n) && (_line_diff(box, line_n) == box->h)) {
        _line_scroll(box);
    }
}

void eos_eve_text_newline(EVEText *box) {
    int line = (box->buf_idx / 2 / box->w + 1) % box->buf_line_h;
    box->buf_idx = line * box->w * 2;
    if (_line_diff(box, line) == box->h) {
        _line_scroll(box);
    }
}