diff options
Diffstat (limited to 'code/fe310/eos')
| -rw-r--r-- | code/fe310/eos/eve/widget/Makefile | 2 | ||||
| -rw-r--r-- | code/fe310/eos/eve/widget/str.c | 282 | ||||
| -rw-r--r-- | code/fe310/eos/eve/widget/str.h | 35 | ||||
| -rw-r--r-- | code/fe310/eos/eve/widget/widget.c | 6 | ||||
| -rw-r--r-- | code/fe310/eos/eve/widget/widget.h | 5 | 
5 files changed, 325 insertions, 5 deletions
| diff --git a/code/fe310/eos/eve/widget/Makefile b/code/fe310/eos/eve/widget/Makefile index 3fe0368..989d9d9 100644 --- a/code/fe310/eos/eve/widget/Makefile +++ b/code/fe310/eos/eve/widget/Makefile @@ -2,7 +2,7 @@ include ../../../common.mk  CFLAGS += -I. -I.. -I../.. -obj = label.o widget.o page.o text.o +obj = label.o widget.o page.o str.o text.o  %.o: %.c %.h diff --git a/code/fe310/eos/eve/widget/str.c b/code/fe310/eos/eve/widget/str.c new file mode 100644 index 0000000..bf86d49 --- /dev/null +++ b/code/fe310/eos/eve/widget/str.c @@ -0,0 +1,282 @@ +#include <stdlib.h> +#include <string.h> +#include <math.h> + +#include "eve.h" +#include "eve_kbd.h" + +#include "screen/screen.h" +#include "screen/window.h" +#include "screen/page.h" +#include "screen/font.h" + +#include "label.h" +#include "widget.h" +#include "str.h" + +#define CH_BS                   0x08 +#define CH_DEL                  0x7f +#define CH_EOF                  0x1a + +#define CH_CTRLX                0x18 +#define CH_CTRLC                0x03 +#define CH_CTRLV                0x16 + +#define STRW_TMODE_CURSOR       1 +#define STRW_TMODE_SCROLL_X     2 +#define STRW_TMODE_SCROLL_Y     3 + +#define CHAR_VALID_INPUT(c)     ((c >= 0x20) && (c < 0x7f)) + +void eve_strw_init(EVEStrWidget *widget, EVERect *g, EVEFont *font, char *str, uint16_t str_size) { + +} + +static EVEStrCursor *cursor_prox(EVEStrWidget *widget, EVEStrCursor *cursor, EVEPage *page, EVETouch *t, short *dx) { +    EVEWidget *_widget = &widget->w; +    int x = eve_page_x(page, t->x0) - _widget->g.x + widget->str_g.x; +    int _dx; + +    *dx = cursor->x - x; +    _dx = *dx < 0 ? -(*dx) : *dx; + +    if (_dx <= widget->font->w) return cursor; +    return NULL; +} + +int eve_strw_touch(EVEWidget *_widget, EVEPage *page, uint8_t tag0, int touch_idx, EVERect *focus) { +    EVEStrWidget *widget = (EVEStrWidget *)_widget; +    EVETouch *t; +    uint16_t evt; + +    if (touch_idx > 0) return 0; + +    t = eve_touch_evt(tag0, touch_idx, widget->tag, 1, &evt); +    if (t && evt) { +        EVEStrCursor *t_cursor = NULL; +        short dx; +        char f = 0; + +        if (evt & (EVE_TOUCH_ETYPE_LPRESS | EVE_TOUCH_ETYPE_TRACK_START)) { +            if (widget->cursor2.on) { +                t_cursor = cursor_prox(widget, &widget->cursor2, page, t, &dx); +            } +            if ((t_cursor == NULL) && widget->cursor1.on) { +                t_cursor = cursor_prox(widget, &widget->cursor1, page, t, &dx); +            } +            if (evt & EVE_TOUCH_ETYPE_TRACK_START) { +                widget->track.cursor = t_cursor; +                if (t_cursor) { +                    widget->track.mode = STRW_TMODE_CURSOR; +                    widget->track.dx = dx; +                } else if (t->eevt & EVE_TOUCH_EETYPE_TRACK_X) { +                    widget->track.mode = STRW_TMODE_SCROLL_X; +                } else if (t->eevt & EVE_TOUCH_EETYPE_TRACK_Y) { +                    widget->track.mode = STRW_TMODE_SCROLL_Y; +                } +            } +        } + +        switch (widget->track.mode) { +            case STRW_TMODE_SCROLL_Y: +                if (page && page->handle_evt) page->handle_evt(page, _widget, t, evt, tag0, touch_idx); +                break; + +            case STRW_TMODE_SCROLL_X: +                if (evt & EVE_TOUCH_ETYPE_TRACK_START) { +                    widget->str_g.x0 = widget->str_g.x; +                } +                if (evt & EVE_TOUCH_ETYPE_TRACK) { +                    int x = widget->str_g.x0 + t->x0 - t->x; +                    int w1 = widget->w.g.w - widget->font->w; + +                    if (x < 0) x = 0; +                    if (x > widget->str_g.w - w1) x = widget->str_g.w - w1; +                    widget->str_g.x = x; +                } +                break; + +            case STRW_TMODE_CURSOR: +                if (evt & EVE_TOUCH_ETYPE_TRACK) eve_strw_cursor_set(widget, widget->track.cursor, eve_page_x(page, t->x) + widget->track.dx); +                break; + +            default: +                if (evt & EVE_TOUCH_ETYPE_LPRESS) { +                    if (widget->cursor2.on) { +                        // copy +                    } else if (widget->cursor1.on) { +                        if (t_cursor) { +                            // paste +                        } else { +                            eve_strw_cursor_set(widget, &widget->cursor2, eve_page_x(page, t->x)); +                        } +                    } else { +                        // select +                    } +                } +                if ((evt & EVE_TOUCH_ETYPE_POINT_UP) && !(t->eevt & EVE_TOUCH_EETYPE_LPRESS)) { +                    eve_strw_cursor_set(widget, &widget->cursor1, eve_page_x(page, t->x)); +                    if (widget->cursor2.on) eve_strw_cursor_clear(widget, &widget->cursor2); +                    f = 1; +                } +                break; +        } + +        if (evt & EVE_TOUCH_ETYPE_TRACK_STOP) { +            widget->track.mode = 0; +            widget->track.cursor = NULL; +            widget->track.dx = 0; +        } + +        if (f && focus) { +            focus->x = _widget->g.x; +            focus->y = _widget->g.y; +            focus->w = _widget->g.w; +            focus->h = 2 * widget->font->h; +        } + +        return 1; +    } + +    return 0; +} + +uint8_t eve_strw_draw(EVEWidget *_widget, EVEPage *page, uint8_t tag0) { + +} + +void eve_strw_putc(void *_page, int c) { +    EVEPage *page = _page; +    EVEStrWidget *widget = (EVEStrWidget *)eve_page_get_focus(page); +    EVEStrCursor *cursor1 = &widget->cursor1; +    EVEStrCursor *cursor2 = &widget->cursor2; +    char *str; +    int w0 = widget->font->w; +    int w1 = widget->w.g.w - widget->font->w; + +    if (c == CH_EOF) { +        if (cursor1->on) eve_strw_cursor_clear(widget, cursor1); +        if (cursor2->on) eve_strw_cursor_clear(widget, cursor2); +        return; +    } + +    if (cursor2->on) { +        EVEStrCursor *c1; +        EVEStrCursor *c2; +        char *clipb = NULL; +        int ins_c = 0, del_c = 0; +        int ins_ch_w = 0, del_ch_w = 0; + +        if (cursor1->ch <= cursor2->ch) { +            c1 = cursor1; +            c2 = cursor2; +        } else { +            c1 = cursor2; +            c2 = cursor1; +        } + +        str = widget->str + c1->ch; +        del_c = c2->ch - c1->ch; +        del_ch_w = eve_font_bufw(widget->font, str, del_c); +        if ((c == CH_CTRLX) || (c == CH_CTRLC)) { +            // eve_clipb_push(str, del_c); +            if (c == CH_CTRLC) return; +        } +        if (CHAR_VALID_INPUT(c) && (widget->str_len < widget->str_size + del_c - 1)) { +            ins_c = 1; +            ins_ch_w = widget->font->w_ch[c]; +        } else if (c == CH_CTRLV) { +            // clipb = eve_clipb_pop(); +            ins_c = clipb ? strlen(clipb) : 0; +            if (ins_c) { +                if (widget->str_len >= widget->str_size - (ins_c - del_c)) ins_c = widget->str_size - widget->str_len + del_c - 1; +                ins_ch_w = eve_font_bufw(widget->font, clipb, ins_c); +            } +        } +        if (ins_c != del_c) memmove(str + ins_c, str + del_c, widget->str_len - c2->ch + 1); +        if (ins_c) { +            if (c == CH_CTRLV) { +                memcpy(str, clipb, ins_c); +            } else { +                *str = c; +            } +            c1->ch += ins_c; +            c1->x += ins_ch_w; +        } +        widget->str_len += ins_c - del_c; +        widget->str_g.w += ins_ch_w - del_ch_w; +        if (c1 == cursor2) widget->cursor1 = widget->cursor2; +        eve_strw_cursor_clear(widget, cursor2); +    } else if (cursor1->on) { +        uint8_t ch_w; + +        str = widget->str + cursor1->ch; +        switch (c) { +            case CH_BS: +                if (cursor1->ch > 0) { +                    ch_w = widget->font->w_ch[*(str - 1)]; +                    memmove(str - 1, str, widget->str_len - cursor1->ch + 1); +                    widget->str_len--; +                    widget->str_g.w -= ch_w; +                    cursor1->ch--; +                    cursor1->x -= ch_w; +                } +                break; + +            case CH_DEL: +                if (cursor1->ch < widget->str_len) { +                    ch_w = widget->font->w_ch[*str]; +                    memmove(str, str + 1, widget->str_len - cursor1->ch); +                    widget->str_len--; +                    widget->str_g.w -= ch_w; +                } +                break; + +            default: +                if (CHAR_VALID_INPUT(c) && (widget->str_len < widget->str_size - 1)) { +                    ch_w = widget->font->w_ch[c]; +                    memmove(str + 1, str, widget->str_len - cursor1->ch + 1); +                    *str = c; +                    widget->str_len++; +                    widget->str_g.w += ch_w; +                    cursor1->ch++; +                    cursor1->x += ch_w; +                } +                break; +        } +    } + +    if (cursor1->x - widget->str_g.x < w0) widget->str_g.x = cursor1->x - w0 > 0 ? cursor1->x - w0 : 0; +    if (cursor1->x - widget->str_g.x > w1) widget->str_g.x = cursor1->x - w1; +} + +void eve_strw_cursor_set(EVEStrWidget *widget, EVEStrCursor *cursor, int16_t x) { +    int i; +    uint16_t _x, _d; +    EVEWidget *_widget = &widget->w; + +    x = x - _widget->g.x + widget->str_g.x; + +    _x = 0; +    _d = x; +    for (i=0; i<widget->str_len; i++) { +        _x += widget->font->w_ch[widget->str[i]]; +        if (_x >= x) { +            if (_x - x < _d) { +                i++; +            } else { +                _x -= widget->font->w_ch[widget->str[i]]; +            } +            break; +        } else { +            _d = x - _x; +        } +    } +    cursor->x = _x; +    cursor->ch = i; +    cursor->on = 1; +} + +void eve_strw_cursor_clear(EVEStrWidget *widget, EVEStrCursor *cursor) { +    cursor->on = 0; +} diff --git a/code/fe310/eos/eve/widget/str.h b/code/fe310/eos/eve/widget/str.h new file mode 100644 index 0000000..04a813e --- /dev/null +++ b/code/fe310/eos/eve/widget/str.h @@ -0,0 +1,35 @@ +#include <stdint.h> + +typedef struct EVEStrCursor { +    char on; +    uint16_t x; +    uint16_t ch; +} EVEStrCursor; + +typedef struct EVEStrWidget { +    EVEWidget w; +    EVEFont *font; +    char *str; +    uint16_t str_size; +    uint16_t str_len; +    struct { +        int16_t x; +        int16_t x0; +        uint16_t w; +    } str_g; +    EVEStrCursor cursor1; +    EVEStrCursor cursor2; +    uint8_t tag; +    struct { +        EVEStrCursor *cursor; +        short dx; +        char mode; +    } track; +} EVEStrWidget; + +void eve_strw_init(EVEStrWidget *widget, EVERect *g, EVEFont *font, char *str, uint16_t str_size); +int eve_strw_touch(EVEWidget *_widget, EVEPage *page, uint8_t tag0, int touch_idx, EVERect *focus); +uint8_t eve_strw_draw(EVEWidget *_widget, EVEPage *page, uint8_t tag0); +void eve_strw_putc(void *_page, int c); +void eve_strw_cursor_set(EVEStrWidget *widget, EVEStrCursor *cursor, int16_t x); +void eve_strw_cursor_clear(EVEStrWidget *widget, EVEStrCursor *cursor);
\ No newline at end of file diff --git a/code/fe310/eos/eve/widget/widget.c b/code/fe310/eos/eve/widget/widget.c index c61a09a..13b784f 100644 --- a/code/fe310/eos/eve/widget/widget.c +++ b/code/fe310/eos/eve/widget/widget.c @@ -13,11 +13,13 @@  #include "widget.h"  #include "page.h"  #include "text.h" +#include "str.h"  static const size_t _eve_wsize[] = {      0, -    sizeof(EVETextWidget), -    sizeof(EVEPageWidget) +    sizeof(EVEPageWidget), +    sizeof(EVEStrWidget), +    sizeof(EVETextWidget)  };  void eve_widget_init(EVEWidget *widget, uint8_t type, EVERect *g, eve_widget_touch_t touch, eve_widget_draw_t draw, eve_kbd_input_handler_t putc) { diff --git a/code/fe310/eos/eve/widget/widget.h b/code/fe310/eos/eve/widget/widget.h index 25e1b59..7083be3 100644 --- a/code/fe310/eos/eve/widget/widget.h +++ b/code/fe310/eos/eve/widget/widget.h @@ -1,7 +1,8 @@  #include <stdint.h> -#define EVE_WIDGET_TYPE_TEXT    1 -#define EVE_WIDGET_TYPE_PAGE    2 +#define EVE_WIDGET_TYPE_PAGE    1 +#define EVE_WIDGET_TYPE_STR     2 +#define EVE_WIDGET_TYPE_TEXT    3  struct EVEWidget; | 
