| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Implementation of a scene, a collection of text/image/menu items in an expo |
| * |
| * Copyright 2022 Google LLC |
| * Written by Simon Glass <sjg@chromium.org> |
| */ |
| |
| #define LOG_CATEGORY LOGC_EXPO |
| |
| #include <alist.h> |
| #include <dm.h> |
| #include <expo.h> |
| #include <malloc.h> |
| #include <mapmem.h> |
| #include <menu.h> |
| #include <video.h> |
| #include <video_console.h> |
| #include <linux/input.h> |
| #include "scene_internal.h" |
| |
| int scene_new(struct expo *exp, const char *name, uint id, struct scene **scnp) |
| { |
| struct scene *scn; |
| |
| scn = calloc(1, sizeof(struct scene)); |
| if (!scn) |
| return log_msg_ret("expo", -ENOMEM); |
| scn->name = strdup(name); |
| if (!scn->name) { |
| free(scn); |
| return log_msg_ret("name", -ENOMEM); |
| } |
| |
| if (!abuf_init_size(&scn->buf, EXPO_MAX_CHARS + 1)) { |
| free(scn->name); |
| free(scn); |
| return log_msg_ret("buf", -ENOMEM); |
| } |
| abuf_init(&scn->entry_save); |
| |
| INIT_LIST_HEAD(&scn->obj_head); |
| scn->id = resolve_id(exp, id); |
| scn->expo = exp; |
| list_add_tail(&scn->sibling, &exp->scene_head); |
| |
| *scnp = scn; |
| |
| return scn->id; |
| } |
| |
| void scene_obj_destroy(struct scene_obj *obj) |
| { |
| if (obj->type == SCENEOBJT_MENU) |
| scene_menu_destroy((struct scene_obj_menu *)obj); |
| free(obj->name); |
| free(obj); |
| } |
| |
| void scene_destroy(struct scene *scn) |
| { |
| struct scene_obj *obj, *next; |
| |
| list_for_each_entry_safe(obj, next, &scn->obj_head, sibling) |
| scene_obj_destroy(obj); |
| |
| abuf_uninit(&scn->entry_save); |
| abuf_uninit(&scn->buf); |
| free(scn->name); |
| free(scn); |
| } |
| |
| int scene_obj_count(struct scene *scn) |
| { |
| return list_count_nodes(&scn->obj_head); |
| } |
| |
| void *scene_obj_find(const struct scene *scn, uint id, enum scene_obj_t type) |
| { |
| struct scene_obj *obj; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| if (obj->id == id && |
| (type == SCENEOBJT_NONE || obj->type == type)) |
| return obj; |
| } |
| |
| return NULL; |
| } |
| |
| void *scene_obj_find_by_name(struct scene *scn, const char *name) |
| { |
| struct scene_obj *obj; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| if (!strcmp(name, obj->name)) |
| return obj; |
| } |
| |
| return NULL; |
| } |
| |
| int scene_obj_add(struct scene *scn, const char *name, uint id, |
| enum scene_obj_t type, uint size, struct scene_obj **objp) |
| { |
| struct scene_obj *obj; |
| |
| obj = calloc(1, size); |
| if (!obj) |
| return log_msg_ret("obj", -ENOMEM); |
| obj->name = strdup(name); |
| if (!obj->name) { |
| free(obj); |
| return log_msg_ret("name", -ENOMEM); |
| } |
| |
| obj->id = resolve_id(scn->expo, id); |
| obj->scene = scn; |
| obj->type = type; |
| list_add_tail(&obj->sibling, &scn->obj_head); |
| *objp = obj; |
| |
| return obj->id; |
| } |
| |
| int scene_img(struct scene *scn, const char *name, uint id, char *data, |
| struct scene_obj_img **imgp) |
| { |
| struct scene_obj_img *img; |
| int ret; |
| |
| ret = scene_obj_add(scn, name, id, SCENEOBJT_IMAGE, |
| sizeof(struct scene_obj_img), |
| (struct scene_obj **)&img); |
| if (ret < 0) |
| return log_msg_ret("obj", ret); |
| |
| img->data = data; |
| |
| if (imgp) |
| *imgp = img; |
| |
| return img->obj.id; |
| } |
| |
| int scene_txt_generic_init(struct expo *exp, struct scene_txt_generic *gen, |
| const char *name, uint str_id, const char *str) |
| { |
| int ret; |
| |
| if (str) { |
| ret = expo_str(exp, name, str_id, str); |
| if (ret < 0) |
| return log_msg_ret("str", ret); |
| if (str_id && ret != str_id) |
| return log_msg_ret("id", -EEXIST); |
| str_id = ret; |
| } else { |
| ret = resolve_id(exp, str_id); |
| if (ret < 0) |
| return log_msg_ret("nst", ret); |
| if (str_id && ret != str_id) |
| return log_msg_ret("nid", -EEXIST); |
| } |
| |
| gen->str_id = str_id; |
| alist_init_struct(&gen->lines, struct vidconsole_mline); |
| |
| return 0; |
| } |
| |
| int scene_txt(struct scene *scn, const char *name, uint id, uint str_id, |
| struct scene_obj_txt **txtp) |
| { |
| struct scene_obj_txt *txt; |
| int ret; |
| |
| ret = scene_obj_add(scn, name, id, SCENEOBJT_TEXT, |
| sizeof(struct scene_obj_txt), |
| (struct scene_obj **)&txt); |
| if (ret < 0) |
| return log_msg_ret("obj", ret); |
| |
| ret = scene_txt_generic_init(scn->expo, &txt->gen, name, str_id, NULL); |
| if (ret) |
| return log_msg_ret("stg", ret); |
| if (txtp) |
| *txtp = txt; |
| |
| return txt->obj.id; |
| } |
| |
| int scene_txt_str(struct scene *scn, const char *name, uint id, uint str_id, |
| const char *str, struct scene_obj_txt **txtp) |
| { |
| struct scene_obj_txt *txt; |
| int ret; |
| |
| ret = scene_obj_add(scn, name, id, SCENEOBJT_TEXT, |
| sizeof(struct scene_obj_txt), |
| (struct scene_obj **)&txt); |
| if (ret < 0) |
| return log_msg_ret("obj", ret); |
| |
| ret = scene_txt_generic_init(scn->expo, &txt->gen, name, str_id, str); |
| if (ret) |
| return log_msg_ret("tsg", ret); |
| if (txtp) |
| *txtp = txt; |
| |
| return txt->obj.id; |
| } |
| |
| int scene_box(struct scene *scn, const char *name, uint id, uint width, |
| struct scene_obj_box **boxp) |
| { |
| struct scene_obj_box *box; |
| int ret; |
| |
| ret = scene_obj_add(scn, name, id, SCENEOBJT_BOX, |
| sizeof(struct scene_obj_box), |
| (struct scene_obj **)&box); |
| if (ret < 0) |
| return log_msg_ret("obj", ret); |
| |
| box->width = width; |
| |
| if (boxp) |
| *boxp = box; |
| |
| return box->obj.id; |
| } |
| |
| int scene_txt_set_font(struct scene *scn, uint id, const char *font_name, |
| uint font_size) |
| { |
| struct scene_obj_txt *txt; |
| |
| txt = scene_obj_find(scn, id, SCENEOBJT_TEXT); |
| if (!txt) |
| return log_msg_ret("find", -ENOENT); |
| txt->gen.font_name = font_name; |
| txt->gen.font_size = font_size; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_pos(struct scene *scn, uint id, int x, int y) |
| { |
| struct scene_obj *obj; |
| int w, h; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| w = obj->bbox.x1 - obj->bbox.x0; |
| h = obj->bbox.y1 - obj->bbox.y0; |
| obj->bbox.x0 = x; |
| obj->bbox.y0 = y; |
| obj->bbox.x1 = obj->bbox.x0 + w; |
| obj->bbox.y1 = obj->bbox.y0 + h; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_size(struct scene *scn, uint id, int w, int h) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| obj->bbox.x1 = obj->bbox.x0 + w; |
| obj->bbox.y1 = obj->bbox.y0 + h; |
| obj->flags |= SCENEOF_SIZE_VALID; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_width(struct scene *scn, uint id, int w) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| obj->bbox.x1 = obj->bbox.x0 + w; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_bbox(struct scene *scn, uint id, int x0, int y0, int x1, |
| int y1) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| obj->bbox.x0 = x0; |
| obj->bbox.y0 = y0; |
| obj->bbox.x1 = x1; |
| obj->bbox.y1 = y1; |
| obj->flags |= SCENEOF_SIZE_VALID; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_halign(struct scene *scn, uint id, enum scene_obj_align aln) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("osh", -ENOENT); |
| obj->horiz = aln; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_valign(struct scene *scn, uint id, enum scene_obj_align aln) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("osv", -ENOENT); |
| obj->vert = aln; |
| |
| return 0; |
| } |
| |
| int scene_obj_set_hide(struct scene *scn, uint id, bool hide) |
| { |
| int ret; |
| |
| ret = scene_obj_flag_clrset(scn, id, SCENEOF_HIDE, |
| hide ? SCENEOF_HIDE : 0); |
| if (ret) |
| return log_msg_ret("flg", ret); |
| |
| return 0; |
| } |
| |
| int scene_obj_flag_clrset(struct scene *scn, uint id, uint clr, uint set) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| obj->flags &= ~clr; |
| obj->flags |= set; |
| |
| return 0; |
| } |
| |
| static void handle_alignment(enum scene_obj_align horiz, |
| enum scene_obj_align vert, |
| struct scene_obj_bbox *bbox, |
| struct scene_obj_dims *dims, |
| int xsize, int ysize, |
| struct scene_obj_offset *offset) |
| { |
| int width, height; |
| |
| if (bbox->x1 == SCENEOB_DISPLAY_MAX) |
| bbox->x1 = xsize ?: 1280; |
| if (bbox->y1 == SCENEOB_DISPLAY_MAX) |
| bbox->y1 = ysize ?: 1024; |
| |
| width = bbox->x1 - bbox->x0; |
| height = bbox->y1 - bbox->y0; |
| |
| switch (horiz) { |
| case SCENEOA_CENTRE: |
| offset->xofs = (width - dims->x) / 2; |
| break; |
| case SCENEOA_RIGHT: |
| offset->xofs = width - dims->x; |
| break; |
| case SCENEOA_LEFT: |
| offset->xofs = 0; |
| break; |
| } |
| |
| switch (vert) { |
| case SCENEOA_CENTRE: |
| offset->yofs = (height - dims->y) / 2; |
| break; |
| case SCENEOA_BOTTOM: |
| offset->yofs = height - dims->y; |
| break; |
| case SCENEOA_TOP: |
| default: |
| offset->yofs = 0; |
| break; |
| } |
| } |
| |
| int scene_obj_get_hw(struct scene *scn, uint id, int *widthp) |
| { |
| struct scene_obj *obj; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_MENU: |
| case SCENEOBJT_TEXTLINE: |
| case SCENEOBJT_BOX: |
| break; |
| case SCENEOBJT_IMAGE: { |
| struct scene_obj_img *img = (struct scene_obj_img *)obj; |
| ulong width, height; |
| uint bpix; |
| |
| video_bmp_get_info(img->data, &width, &height, &bpix); |
| if (widthp) |
| *widthp = width; |
| return height; |
| } |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_TEXTEDIT: { |
| struct scene_txt_generic *gen; |
| struct expo *exp = scn->expo; |
| struct vidconsole_bbox bbox; |
| int len, ret, limit; |
| const char *str; |
| |
| if (obj->type == SCENEOBJT_TEXT) |
| gen = &((struct scene_obj_txt *)obj)->gen; |
| else |
| gen = &((struct scene_obj_txtedit *)obj)->gen; |
| |
| str = expo_get_str(exp, gen->str_id); |
| if (!str) |
| return log_msg_ret("str", -ENOENT); |
| len = strlen(str); |
| |
| /* if there is no console, make it up */ |
| if (!exp->cons) { |
| if (widthp) |
| *widthp = 8 * len; |
| return 16; |
| } |
| |
| limit = obj->flags & SCENEOF_SIZE_VALID ? |
| obj->bbox.x1 - obj->bbox.x0 : -1; |
| |
| ret = vidconsole_measure(scn->expo->cons, gen->font_name, |
| gen->font_size, str, limit, &bbox, |
| &gen->lines); |
| if (ret) |
| return log_msg_ret("mea", ret); |
| if (widthp) |
| *widthp = bbox.x1; |
| |
| return bbox.y1; |
| } |
| } |
| |
| return 0; |
| } |
| |
| /** |
| * scene_render_background() - Render the background for an object |
| * |
| * @obj: Object to render |
| * @box_only: true to show a box around the object, but keep the normal |
| * background colour inside |
| * @cur_item: true to render the background only for the current menu item |
| */ |
| static void scene_render_background(struct scene_obj *obj, bool box_only, |
| bool cur_item) |
| { |
| struct vidconsole_bbox bbox[SCENEBB_count], *sel; |
| struct expo *exp = obj->scene->expo; |
| const struct expo_theme *theme = &exp->theme; |
| struct udevice *dev = exp->display; |
| struct video_priv *vid_priv; |
| struct udevice *cons = exp->cons; |
| struct vidconsole_colour old; |
| enum colour_idx fore, back; |
| uint inset = theme->menu_inset; |
| |
| vid_priv = dev_get_uclass_priv(dev); |
| /* draw a background for the object */ |
| if (vid_priv->white_on_black) { |
| fore = VID_DARK_GREY; |
| back = VID_WHITE; |
| } else { |
| fore = VID_LIGHT_GRAY; |
| back = VID_BLACK; |
| } |
| |
| /* see if this object wants to render a background */ |
| if (scene_obj_calc_bbox(obj, bbox)) |
| return; |
| |
| sel = cur_item ? &bbox[SCENEBB_curitem] : &bbox[SCENEBB_label]; |
| if (!sel->valid) |
| return; |
| |
| vidconsole_push_colour(cons, fore, back, &old); |
| video_fill_part(dev, sel->x0 - inset, sel->y0 - inset, |
| sel->x1 + inset, sel->y1 + inset, |
| vid_priv->colour_fg); |
| vidconsole_pop_colour(cons, &old); |
| if (box_only) { |
| video_fill_part(dev, sel->x0, sel->y0, sel->x1, sel->y1, |
| vid_priv->colour_bg); |
| } |
| } |
| |
| static int scene_txt_render(struct expo *exp, struct udevice *dev, |
| struct udevice *cons, struct scene_obj *obj, |
| struct scene_txt_generic *gen, int x, int y, |
| int menu_inset) |
| { |
| const struct vidconsole_mline *mline, *last; |
| struct video_priv *vid_priv; |
| struct vidconsole_colour old; |
| enum colour_idx fore, back; |
| struct scene_obj_dims dims; |
| struct scene_obj_bbox bbox; |
| const char *str; |
| int ret; |
| |
| if (!cons) |
| return -ENOTSUPP; |
| |
| if (gen->font_name || gen->font_size) { |
| ret = vidconsole_select_font(cons, gen->font_name, |
| gen->font_size); |
| } else { |
| ret = vidconsole_select_font(cons, NULL, 0); |
| } |
| if (ret && ret != -ENOSYS) |
| return log_msg_ret("font", ret); |
| str = expo_get_str(exp, gen->str_id); |
| if (!str) |
| return 0; |
| |
| vid_priv = dev_get_uclass_priv(dev); |
| if (vid_priv->white_on_black) { |
| fore = VID_BLACK; |
| back = VID_WHITE; |
| } else { |
| fore = VID_LIGHT_GRAY; |
| back = VID_BLACK; |
| } |
| |
| if (obj->flags & SCENEOF_POINT) { |
| int inset; |
| |
| inset = exp->popup ? menu_inset : 0; |
| vidconsole_push_colour(cons, fore, back, &old); |
| video_fill_part(dev, x - inset, y, |
| obj->bbox.x1, obj->bbox.y1, |
| vid_priv->colour_bg); |
| } |
| |
| mline = alist_get(&gen->lines, 0, typeof(*mline)); |
| last = alist_get(&gen->lines, gen->lines.count - 1, typeof(*mline)); |
| if (mline) |
| dims.y = last->bbox.y1 - mline->bbox.y0; |
| bbox.y0 = obj->bbox.y0; |
| bbox.y1 = obj->bbox.y1; |
| |
| if (!mline) { |
| vidconsole_set_cursor_pos(cons, x, y); |
| vidconsole_put_string(cons, str); |
| } |
| |
| alist_for_each(mline, &gen->lines) { |
| struct scene_obj_offset offset; |
| |
| bbox.x0 = obj->bbox.x0; |
| bbox.x1 = obj->bbox.x1; |
| dims.x = mline->bbox.x1 - mline->bbox.x0; |
| handle_alignment(obj->horiz, obj->vert, &bbox, &dims, |
| obj->bbox.x1 - obj->bbox.x0, |
| obj->bbox.y1 - obj->bbox.y0, &offset); |
| |
| x = obj->bbox.x0 + offset.xofs; |
| y = obj->bbox.y0 + offset.yofs + mline->bbox.y0; |
| if (y > bbox.y1) |
| break; /* clip this line and any following */ |
| vidconsole_set_cursor_pos(cons, x, y); |
| vidconsole_put_stringn(cons, str + mline->start, mline->len); |
| } |
| if (obj->flags & SCENEOF_POINT) |
| vidconsole_pop_colour(cons, &old); |
| |
| return 0; |
| } |
| |
| /** |
| * scene_obj_render() - Render an object |
| * |
| * @obj: Object to render |
| * @text_mode: true to use text mode |
| * Return: 0 if OK, -ve on error |
| */ |
| static int scene_obj_render(struct scene_obj *obj, bool text_mode) |
| { |
| struct scene *scn = obj->scene; |
| struct expo *exp = scn->expo; |
| const struct expo_theme *theme = &exp->theme; |
| struct udevice *dev = exp->display; |
| struct udevice *cons = text_mode ? NULL : exp->cons; |
| struct video_priv *vid_priv; |
| int x, y, ret; |
| |
| y = obj->bbox.y0; |
| x = obj->bbox.x0 + obj->ofs.xofs; |
| vid_priv = dev_get_uclass_priv(dev); |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| break; |
| case SCENEOBJT_IMAGE: { |
| struct scene_obj_img *img = (struct scene_obj_img *)obj; |
| |
| if (!cons) |
| return -ENOTSUPP; |
| ret = video_bmp_display(dev, map_to_sysmem(img->data), x, y, |
| true); |
| if (ret < 0) |
| return log_msg_ret("img", ret); |
| break; |
| } |
| case SCENEOBJT_TEXT: { |
| struct scene_obj_txt *txt = (struct scene_obj_txt *)obj; |
| |
| ret = scene_txt_render(exp, dev, cons, obj, &txt->gen, x, y, |
| theme->menu_inset); |
| break; |
| } |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu = (struct scene_obj_menu *)obj; |
| |
| if (exp->popup) { |
| if (obj->flags & SCENEOF_OPEN) { |
| if (!cons) |
| return -ENOTSUPP; |
| |
| /* draw a background behind the menu items */ |
| scene_render_background(obj, false, false); |
| } |
| } else if (exp->show_highlight) { |
| /* do nothing */ |
| } |
| |
| /* |
| * With a vidconsole, the text and item pointer are rendered as |
| * normal objects so we don't need to do anything here. The menu |
| * simply controls where they are positioned. |
| */ |
| if (cons) |
| return -ENOTSUPP; |
| |
| ret = scene_menu_display(menu); |
| if (ret < 0) |
| return log_msg_ret("img", ret); |
| |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: |
| if (obj->flags & SCENEOF_OPEN) |
| scene_render_background(obj, true, false); |
| break; |
| case SCENEOBJT_BOX: { |
| struct scene_obj_box *box = (struct scene_obj_box *)obj; |
| |
| video_draw_box(dev, obj->bbox.x0, obj->bbox.y0, obj->bbox.x1, |
| obj->bbox.y1, box->width, vid_priv->colour_fg); |
| break; |
| } |
| case SCENEOBJT_TEXTEDIT: { |
| struct scene_obj_txtedit *ted = (struct scene_obj_txtedit *)obj; |
| |
| ret = scene_txt_render(exp, dev, cons, obj, &ted->gen, x, y, |
| theme->menu_inset); |
| break; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_calc_arrange(struct scene *scn, struct expo_arrange_info *arr) |
| { |
| struct scene_obj *obj; |
| |
| arr->label_width = 0; |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| uint label_id = 0; |
| int width; |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| break; |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu; |
| |
| menu = (struct scene_obj_menu *)obj, |
| label_id = menu->title_id; |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: { |
| struct scene_obj_textline *tline; |
| |
| tline = (struct scene_obj_textline *)obj, |
| label_id = tline->label_id; |
| break; |
| } |
| } |
| |
| if (label_id) { |
| int ret; |
| |
| ret = scene_obj_get_hw(scn, label_id, &width); |
| if (ret < 0) |
| return log_msg_ret("hei", ret); |
| arr->label_width = max(arr->label_width, width); |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_arrange(struct scene *scn) |
| { |
| struct expo_arrange_info arr; |
| int xsize = 0, ysize = 0; |
| struct scene_obj *obj; |
| struct udevice *dev; |
| int ret; |
| |
| dev = scn->expo->display; |
| if (dev) { |
| struct video_priv *priv = dev_get_uclass_priv(dev); |
| |
| xsize = priv->xsize; |
| ysize = priv->ysize; |
| } |
| |
| ret = scene_calc_arrange(scn, &arr); |
| if (ret < 0) |
| return log_msg_ret("arr", ret); |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| handle_alignment(obj->horiz, obj->vert, &obj->bbox, &obj->dims, |
| xsize, ysize, &obj->ofs); |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| break; |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu; |
| |
| menu = (struct scene_obj_menu *)obj, |
| ret = scene_menu_arrange(scn, &arr, menu); |
| if (ret) |
| return log_msg_ret("arr", ret); |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: { |
| struct scene_obj_textline *tline; |
| |
| tline = (struct scene_obj_textline *)obj, |
| ret = scene_textline_arrange(scn, &arr, tline); |
| if (ret) |
| return log_msg_ret("arr", ret); |
| break; |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_render_deps(struct scene *scn, uint id) |
| { |
| struct scene_obj *obj; |
| int ret; |
| |
| if (!id) |
| return 0; |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("obj", -ENOENT); |
| |
| if (!(obj->flags & SCENEOF_HIDE)) { |
| ret = scene_obj_render(obj, false); |
| if (ret && ret != -ENOTSUPP) |
| return log_msg_ret("ren", ret); |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| break; |
| case SCENEOBJT_MENU: |
| scene_menu_render_deps(scn, |
| (struct scene_obj_menu *)obj); |
| break; |
| case SCENEOBJT_TEXTLINE: |
| scene_textline_render_deps(scn, |
| (struct scene_obj_textline *)obj); |
| break; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_render(struct scene *scn) |
| { |
| struct expo *exp = scn->expo; |
| struct scene_obj *obj; |
| int ret; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| if (!(obj->flags & SCENEOF_HIDE)) { |
| ret = scene_obj_render(obj, exp->text_mode); |
| if (ret && ret != -ENOTSUPP) |
| return log_msg_ret("ren", ret); |
| } |
| } |
| |
| /* render any highlighted object on top of the others */ |
| if (scn->highlight_id && !exp->text_mode) { |
| ret = scene_render_deps(scn, scn->highlight_id); |
| if (ret && ret != -ENOTSUPP) |
| return log_msg_ret("dep", ret); |
| } |
| |
| return 0; |
| } |
| |
| /** |
| * send_key_obj() - Handle a keypress for moving between objects |
| * |
| * @scn: Scene to receive the key |
| * @key: Key to send (KEYCODE_UP) |
| * @event: Returns resulting event from this keypress |
| * Returns: 0 if OK, -ve on error |
| */ |
| static void send_key_obj(struct scene *scn, struct scene_obj *obj, int key, |
| struct expo_action *event) |
| { |
| switch (key) { |
| case BKEY_UP: |
| while (obj != list_first_entry(&scn->obj_head, struct scene_obj, |
| sibling)) { |
| obj = list_entry(obj->sibling.prev, |
| struct scene_obj, sibling); |
| if (scene_obj_can_highlight(obj)) { |
| event->type = EXPOACT_POINT_OBJ; |
| event->select.id = obj->id; |
| log_debug("up to obj %d\n", event->select.id); |
| break; |
| } |
| } |
| break; |
| case BKEY_DOWN: |
| while (!list_is_last(&obj->sibling, &scn->obj_head)) { |
| obj = list_entry(obj->sibling.next, struct scene_obj, |
| sibling); |
| if (scene_obj_can_highlight(obj)) { |
| event->type = EXPOACT_POINT_OBJ; |
| event->select.id = obj->id; |
| log_debug("down to obj %d\n", event->select.id); |
| break; |
| } |
| } |
| break; |
| case BKEY_SELECT: |
| if (scene_obj_can_highlight(obj)) { |
| event->type = EXPOACT_OPEN; |
| event->select.id = obj->id; |
| log_debug("open obj %d\n", event->select.id); |
| } |
| break; |
| case BKEY_QUIT: |
| event->type = EXPOACT_QUIT; |
| log_debug("obj quit\n"); |
| break; |
| } |
| } |
| |
| int scene_send_key(struct scene *scn, int key, struct expo_action *event) |
| { |
| struct scene_obj *obj; |
| int ret; |
| |
| event->type = EXPOACT_NONE; |
| |
| /* |
| * In 'popup' mode, arrow keys move betwen objects, unless a menu is |
| * opened |
| */ |
| if (scn->expo->popup) { |
| obj = NULL; |
| if (scn->highlight_id) { |
| obj = scene_obj_find(scn, scn->highlight_id, |
| SCENEOBJT_NONE); |
| } |
| if (!obj) |
| return 0; |
| |
| if (!(obj->flags & SCENEOF_OPEN)) { |
| send_key_obj(scn, obj, key, event); |
| return 0; |
| } |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| break; |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu; |
| |
| menu = (struct scene_obj_menu *)obj, |
| ret = scene_menu_send_key(scn, menu, key, event); |
| if (ret) |
| return log_msg_ret("key", ret); |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: { |
| struct scene_obj_textline *tline; |
| |
| tline = (struct scene_obj_textline *)obj, |
| ret = scene_textline_send_key(scn, tline, key, event); |
| if (ret) |
| return log_msg_ret("key", ret); |
| break; |
| } |
| case SCENEOBJT_TEXTEDIT: |
| /* TODO(sjg@chromium.org): Implement this */ |
| break; |
| } |
| return 0; |
| } |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| if (obj->type == SCENEOBJT_MENU) { |
| struct scene_obj_menu *menu; |
| |
| menu = (struct scene_obj_menu *)obj, |
| ret = scene_menu_send_key(scn, menu, key, event); |
| if (ret) |
| return log_msg_ret("key", ret); |
| break; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_obj_calc_bbox(struct scene_obj *obj, struct vidconsole_bbox bbox[]) |
| { |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| return -ENOSYS; |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu = (struct scene_obj_menu *)obj; |
| |
| scene_menu_calc_bbox(menu, bbox); |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: { |
| struct scene_obj_textline *tline; |
| |
| tline = (struct scene_obj_textline *)obj; |
| scene_textline_calc_bbox(tline, &bbox[SCENEBB_all], |
| &bbox[SCENEBB_label]); |
| break; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_calc_dims(struct scene *scn, bool do_menus) |
| { |
| struct scene_obj *obj; |
| int ret; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| case SCENEOBJT_IMAGE: { |
| int width; |
| |
| if (!do_menus) { |
| ret = scene_obj_get_hw(scn, obj->id, &width); |
| if (ret < 0) |
| return log_msg_ret("get", ret); |
| obj->dims.x = width; |
| obj->dims.y = ret; |
| if (!(obj->flags & SCENEOF_SIZE_VALID)) { |
| obj->bbox.x1 = obj->bbox.x0 + width; |
| obj->bbox.y1 = obj->bbox.y0 + ret; |
| obj->flags |= SCENEOF_SIZE_VALID; |
| } |
| } |
| break; |
| } |
| case SCENEOBJT_MENU: { |
| struct scene_obj_menu *menu; |
| |
| if (do_menus) { |
| menu = (struct scene_obj_menu *)obj; |
| |
| ret = scene_menu_calc_dims(menu); |
| if (ret) |
| return log_msg_ret("men", ret); |
| } |
| break; |
| } |
| case SCENEOBJT_TEXTLINE: { |
| struct scene_obj_textline *tline; |
| |
| tline = (struct scene_obj_textline *)obj; |
| ret = scene_textline_calc_dims(tline); |
| if (ret) |
| return log_msg_ret("men", ret); |
| |
| break; |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| int scene_apply_theme(struct scene *scn, struct expo_theme *theme) |
| { |
| struct scene_obj *obj; |
| int ret; |
| |
| /* Avoid error-checking optional items */ |
| scene_txt_set_font(scn, scn->title_id, NULL, theme->font_size); |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_MENU: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTLINE: |
| break; |
| case SCENEOBJT_TEXTEDIT: |
| scene_txted_set_font(scn, obj->id, NULL, |
| theme->font_size); |
| break; |
| case SCENEOBJT_TEXT: |
| scene_txt_set_font(scn, obj->id, NULL, |
| theme->font_size); |
| break; |
| } |
| } |
| |
| ret = scene_arrange(scn); |
| if (ret) |
| return log_msg_ret("arr", ret); |
| |
| return 0; |
| } |
| |
| void scene_set_highlight_id(struct scene *scn, uint id) |
| { |
| scn->highlight_id = id; |
| } |
| |
| void scene_highlight_first(struct scene *scn) |
| { |
| struct scene_obj *obj; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| if (scene_obj_can_highlight(obj)) { |
| scene_set_highlight_id(scn, obj->id); |
| return; |
| } |
| } |
| } |
| |
| static int scene_obj_open(struct scene *scn, struct scene_obj *obj) |
| { |
| int ret; |
| |
| switch (obj->type) { |
| case SCENEOBJT_NONE: |
| case SCENEOBJT_IMAGE: |
| case SCENEOBJT_MENU: |
| case SCENEOBJT_TEXT: |
| case SCENEOBJT_BOX: |
| case SCENEOBJT_TEXTEDIT: |
| break; |
| case SCENEOBJT_TEXTLINE: |
| ret = scene_textline_open(scn, |
| (struct scene_obj_textline *)obj); |
| if (ret) |
| return log_msg_ret("op", ret); |
| break; |
| } |
| |
| return 0; |
| } |
| |
| int scene_set_open(struct scene *scn, uint id, bool open) |
| { |
| struct scene_obj *obj; |
| int ret; |
| |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("find", -ENOENT); |
| |
| if (open) { |
| ret = scene_obj_open(scn, obj); |
| if (ret) |
| return log_msg_ret("op", ret); |
| } |
| |
| ret = scene_obj_flag_clrset(scn, id, SCENEOF_OPEN, |
| open ? SCENEOF_OPEN : 0); |
| if (ret) |
| return log_msg_ret("flg", ret); |
| |
| return 0; |
| } |
| |
| int scene_iter_objs(struct scene *scn, expo_scene_obj_iterator iter, |
| void *priv) |
| { |
| struct scene_obj *obj; |
| |
| list_for_each_entry(obj, &scn->obj_head, sibling) { |
| int ret; |
| |
| ret = iter(obj, priv); |
| if (ret) |
| return log_msg_ret("itr", ret); |
| } |
| |
| return 0; |
| } |
| |
| int scene_bbox_join(const struct vidconsole_bbox *src, int inset, |
| struct vidconsole_bbox *dst) |
| { |
| if (dst->valid) { |
| dst->x0 = min(dst->x0, src->x0 - inset); |
| dst->y0 = min(dst->y0, src->y0); |
| dst->x1 = max(dst->x1, src->x1 + inset); |
| dst->y1 = max(dst->y1, src->y1); |
| } else { |
| dst->x0 = src->x0 - inset; |
| dst->y0 = src->y0; |
| dst->x1 = src->x1 + inset; |
| dst->y1 = src->y1; |
| dst->valid = true; |
| } |
| |
| return 0; |
| } |
| |
| int scene_bbox_union(struct scene *scn, uint id, int inset, |
| struct vidconsole_bbox *bbox) |
| { |
| struct scene_obj *obj; |
| struct vidconsole_bbox local; |
| |
| if (!id) |
| return 0; |
| obj = scene_obj_find(scn, id, SCENEOBJT_NONE); |
| if (!obj) |
| return log_msg_ret("obj", -ENOENT); |
| local.x0 = obj->bbox.x0; |
| local.y0 = obj->bbox.y0; |
| local.x1 = obj->bbox.x1; |
| local.y1 = obj->bbox.y1; |
| local.valid = true; |
| scene_bbox_join(&local, inset, bbox); |
| |
| return 0; |
| } |