gui_list.c 30.3 KB
Newer Older
Colin D Bennett's avatar
Colin D Bennett committed
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
/* gui_list.c - GUI component to display a selectable list of items.  */
/*
 *  GRUB  --  GRand Unified Bootloader
 *  Copyright (C) 2008,2009  Free Software Foundation, Inc.
 *
 *  GRUB is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <grub/mm.h>
#include <grub/misc.h>
#include <grub/gui.h>
#include <grub/gui_string_util.h>
#include <grub/gfxmenu_view.h>
#include <grub/gfxwidgets.h>
26
#include <grub/color.h>
Colin D Bennett's avatar
Colin D Bennett committed
27

28 29 30 31 32 33
enum scrollbar_slice_mode {
  SCROLLBAR_SLICE_WEST,
  SCROLLBAR_SLICE_CENTER,
  SCROLLBAR_SLICE_EAST
};

Colin D Bennett's avatar
Colin D Bennett committed
34 35
struct grub_gui_list_impl
{
36
  struct grub_gui_list list;
Colin D Bennett's avatar
Colin D Bennett committed
37 38 39 40 41 42 43 44 45 46 47 48 49

  grub_gui_container_t parent;
  grub_video_rect_t bounds;
  char *id;
  int visible;

  int icon_width;
  int icon_height;
  int item_height;
  int item_padding;
  int item_icon_space;
  int item_spacing;
  grub_font_t item_font;
50
  int selected_item_font_inherit;
Colin D Bennett's avatar
Colin D Bennett committed
51
  grub_font_t selected_item_font;
52
  grub_video_rgba_color_t item_color;
53
  int selected_item_color_inherit;
54
  grub_video_rgba_color_t selected_item_color;
Colin D Bennett's avatar
Colin D Bennett committed
55 56 57 58 59 60 61

  int draw_scrollbar;
  int need_to_recreate_scrollbar;
  char *scrollbar_frame_pattern;
  char *scrollbar_thumb_pattern;
  grub_gfxmenu_box_t scrollbar_frame;
  grub_gfxmenu_box_t scrollbar_thumb;
62
  int scrollbar_thumb_overlay;
Colin D Bennett's avatar
Colin D Bennett committed
63
  int scrollbar_width;
64
  enum scrollbar_slice_mode scrollbar_slice;
65 66 67 68
  int scrollbar_left_pad;
  int scrollbar_right_pad;
  int scrollbar_top_pad;
  int scrollbar_bottom_pad;
Colin D Bennett's avatar
Colin D Bennett committed
69 70 71 72 73 74

  int first_shown_index;

  int need_to_recreate_boxes;
  char *theme_dir;
  char *menu_box_pattern;
75 76
  char *item_box_pattern;
  int selected_item_box_pattern_inherit;
Colin D Bennett's avatar
Colin D Bennett committed
77 78 79
  char *selected_item_box_pattern;
  grub_gfxmenu_box_t menu_box;
  grub_gfxmenu_box_t selected_item_box;
80
  grub_gfxmenu_box_t item_box;
Colin D Bennett's avatar
Colin D Bennett committed
81 82

  grub_gfxmenu_icon_manager_t icon_manager;
83 84

  grub_gfxmenu_view_t view;
Colin D Bennett's avatar
Colin D Bennett committed
85 86 87 88 89 90 91 92 93 94 95
};

typedef struct grub_gui_list_impl *list_impl_t;

static void
list_destroy (void *vself)
{
  list_impl_t self = vself;

  grub_free (self->theme_dir);
  grub_free (self->menu_box_pattern);
96
  grub_free (self->item_box_pattern);
Colin D Bennett's avatar
Colin D Bennett committed
97 98 99
  grub_free (self->selected_item_box_pattern);
  if (self->menu_box)
    self->menu_box->destroy (self->menu_box);
100 101
  if (self->item_box)
    self->item_box->destroy (self->item_box);
Colin D Bennett's avatar
Colin D Bennett committed
102 103 104 105
  if (self->selected_item_box)
    self->selected_item_box->destroy (self->selected_item_box);
  if (self->icon_manager)
    grub_gfxmenu_icon_manager_destroy (self->icon_manager);
106 107 108 109
  if (self->scrollbar_thumb)
    self->scrollbar_thumb->destroy (self->scrollbar_thumb);
  if (self->scrollbar_frame)
    self->scrollbar_frame->destroy (self->scrollbar_frame);
110
  grub_free (self->scrollbar_thumb_pattern);
111
  grub_free (self->scrollbar_frame_pattern);
Colin D Bennett's avatar
Colin D Bennett committed
112 113 114 115 116 117
  grub_free (self);
}

static int
get_num_shown_items (list_impl_t self)
{
118 119 120 121 122 123 124
  int boxpad = self->item_padding;
  int item_vspace = self->item_spacing;
  int item_height = self->item_height;
  
  grub_gfxmenu_box_t box = self->menu_box;
  int box_top_pad = box->get_top_pad (box);
  int box_bottom_pad = box->get_bottom_pad (box);
125
  grub_gfxmenu_box_t itembox = self->item_box;
126
  grub_gfxmenu_box_t selbox = self->selected_item_box;
127 128
  int item_top_pad = itembox->get_top_pad (itembox);
  int item_bottom_pad = itembox->get_bottom_pad (itembox);
129 130
  int sel_top_pad = selbox->get_top_pad (selbox);
  int sel_bottom_pad = selbox->get_bottom_pad (selbox);
131 132 133
  int max_top_pad = grub_max (item_top_pad, sel_top_pad);
  int max_bottom_pad = grub_max (item_bottom_pad, sel_bottom_pad);

134 135 136
  if (item_height + item_vspace <= 0)
    return 1;

137
  return (self->bounds.height + item_vspace - 2 * boxpad
138
          - max_top_pad - max_bottom_pad
139
          - box_top_pad - box_bottom_pad) / (item_height + item_vspace);
Colin D Bennett's avatar
Colin D Bennett committed
140 141 142 143 144 145 146 147 148 149 150
}

static int
check_boxes (list_impl_t self)
{
  if (self->need_to_recreate_boxes)
    {
      grub_gui_recreate_box (&self->menu_box,
                             self->menu_box_pattern,
                             self->theme_dir);

151 152 153 154
      grub_gui_recreate_box (&self->item_box,
                             self->item_box_pattern,
                             self->theme_dir);

Colin D Bennett's avatar
Colin D Bennett committed
155 156 157 158 159 160 161
      grub_gui_recreate_box (&self->selected_item_box,
                             self->selected_item_box_pattern,
                             self->theme_dir);

      self->need_to_recreate_boxes = 0;
    }

162 163
  return (self->menu_box != 0 && self->selected_item_box != 0
          && self->item_box != 0);
Colin D Bennett's avatar
Colin D Bennett committed
164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181
}

static int
check_scrollbar (list_impl_t self)
{
  if (self->need_to_recreate_scrollbar)
    {
      grub_gui_recreate_box (&self->scrollbar_frame,
                             self->scrollbar_frame_pattern,
                             self->theme_dir);

      grub_gui_recreate_box (&self->scrollbar_thumb,
                             self->scrollbar_thumb_pattern,
                             self->theme_dir);

      self->need_to_recreate_scrollbar = 0;
    }

182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209
  if (self->scrollbar_frame == 0 || self->scrollbar_thumb == 0)
    return 0;

  /* Sanity checks. */
  grub_gfxmenu_box_t frame = self->scrollbar_frame;
  grub_gfxmenu_box_t thumb = self->scrollbar_thumb;
  grub_gfxmenu_box_t menu = self->menu_box;
  int min_width = frame->get_left_pad (frame)
                  + frame->get_right_pad (frame);
  int min_height = frame->get_top_pad (frame)
                   + frame->get_bottom_pad (frame)
                   + self->scrollbar_top_pad + self->scrollbar_bottom_pad
                   + menu->get_top_pad (menu)
                   + menu->get_bottom_pad (menu);
  if (!self->scrollbar_thumb_overlay)
    {
      min_width += thumb->get_left_pad (thumb)
                   + thumb->get_right_pad (thumb);
      min_height += thumb->get_top_pad (thumb)
                    + thumb->get_bottom_pad (thumb);
    }
  if (min_width <= self->scrollbar_width
      && min_height <= (int) self->bounds.height)
    return 1;

  /* Unprintable dimenstions. */
  self->draw_scrollbar = 0;
  return 0;
Colin D Bennett's avatar
Colin D Bennett committed
210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229
}

static const char *
list_get_id (void *vself)
{
  list_impl_t self = vself;
  return self->id;
}

static int
list_is_instance (void *vself __attribute__((unused)), const char *type)
{
  return (grub_strcmp (type, "component") == 0
          || grub_strcmp (type, "list") == 0);
}

static struct grub_video_bitmap *
get_item_icon (list_impl_t self, int item_index)
{
  grub_menu_entry_t entry;
230
  entry = grub_menu_get_entry (self->view->menu, item_index);
Colin D Bennett's avatar
Colin D Bennett committed
231 232 233 234 235 236 237 238 239
  if (! entry)
    return 0;

  return grub_gfxmenu_icon_manager_get_icon (self->icon_manager, entry);
}

static void
make_selected_item_visible (list_impl_t self)
{
240
  int selected_index = self->view->selected;
Colin D Bennett's avatar
Colin D Bennett committed
241 242 243 244 245 246 247 248 249 250 251 252 253 254
  if (selected_index < 0)
    return;   /* No item is selected.  */
  int num_shown_items = get_num_shown_items (self);
  int last_shown_index = self->first_shown_index + (num_shown_items - 1);
  if (selected_index < self->first_shown_index)
    self->first_shown_index = selected_index;
  else if (selected_index > last_shown_index)
    self->first_shown_index = selected_index - (num_shown_items - 1);
}

/* Draw a scrollbar on the menu.  */
static void
draw_scrollbar (list_impl_t self,
                int value, int extent, int min, int max,
255
                int scrollbar_width, int scrollbar_height)
Colin D Bennett's avatar
Colin D Bennett committed
256
{
257 258
  unsigned thumby, thumbheight;

Colin D Bennett's avatar
Colin D Bennett committed
259 260 261 262 263 264
  grub_gfxmenu_box_t frame = self->scrollbar_frame;
  grub_gfxmenu_box_t thumb = self->scrollbar_thumb;
  int frame_vertical_pad = (frame->get_top_pad (frame)
                            + frame->get_bottom_pad (frame));
  int frame_horizontal_pad = (frame->get_left_pad (frame)
                              + frame->get_right_pad (frame));
265 266
  unsigned thumb_vertical_pad = (thumb->get_top_pad (thumb)
				 + thumb->get_bottom_pad (thumb));
267 268 269
  int thumb_horizontal_pad = (thumb->get_left_pad (thumb)
                              + thumb->get_right_pad (thumb));
  int tracktop = frame->get_top_pad (frame);
270 271 272 273 274
  unsigned tracklen;
  if (scrollbar_height <= frame_vertical_pad)
    tracklen = 0;
  else
    tracklen = scrollbar_height - frame_vertical_pad;
275 276 277
  frame->set_content_size (frame,
                           scrollbar_width - frame_horizontal_pad,
                           tracklen);
278 279 280 281 282
  if (self->scrollbar_thumb_overlay)
    {
      tracklen += thumb_vertical_pad;
      tracktop -= thumb->get_top_pad (thumb);
    }
283 284 285 286 287 288 289 290 291 292
  if (value <= min || max <= min)
    thumby = 0;
  else
    thumby = ((unsigned) tracklen * (value - min))
      / ((unsigned) (max - min));
  if (max <= min)
    thumbheight = 1;
  else
    thumbheight = ((unsigned) (tracklen * extent)
		   / ((unsigned) (max - min))) + 1;
293 294 295 296
  /* Rare occasion: too many entries or too low height. */
  if (thumbheight < thumb_vertical_pad)
    {
      thumbheight = thumb_vertical_pad;
297 298 299 300 301 302
      if (value <= min || max <= extent
	  || tracklen <= thumb_vertical_pad)
	thumby = 0;
      else
	thumby = ((unsigned) ((tracklen - thumb_vertical_pad) * (value - min))
		  / ((unsigned)(max - extent)));
303
    }
304
  thumby += tracktop;
305 306 307 308 309 310 311
  int thumbx = frame->get_left_pad (frame);
  int thumbwidth = scrollbar_width - frame_horizontal_pad;
  if (!self->scrollbar_thumb_overlay)
    thumbwidth -= thumb_horizontal_pad;
  else
    thumbx -= thumb->get_left_pad (thumb);
  thumb->set_content_size (thumb, thumbwidth,
312 313
                           thumbheight - thumb_vertical_pad);
  frame->draw (frame, 0, 0);
314
  thumb->draw (thumb, thumbx, thumby);
Colin D Bennett's avatar
Colin D Bennett committed
315 316 317 318
}

/* Draw the list of items.  */
static void
319
draw_menu (list_impl_t self, int num_shown_items)
Colin D Bennett's avatar
Colin D Bennett committed
320
{
321
  if (! self->menu_box || ! self->selected_item_box || ! self->item_box)
Colin D Bennett's avatar
Colin D Bennett committed
322 323 324 325 326 327 328 329
    return;

  int boxpad = self->item_padding;
  int icon_text_space = self->item_icon_space;
  int item_vspace = self->item_spacing;

  int ascent = grub_font_get_ascent (self->item_font);
  int descent = grub_font_get_descent (self->item_font);
330 331
  int selected_ascent = grub_font_get_ascent (self->selected_item_font);
  int selected_descent = grub_font_get_descent (self->selected_item_font);
332
  int text_box_height = self->item_height;
Colin D Bennett's avatar
Colin D Bennett committed
333 334 335

  make_selected_item_visible (self);

336
  grub_gfxmenu_box_t itembox = self->item_box;
337
  grub_gfxmenu_box_t selbox = self->selected_item_box;
338 339 340 341
  int item_leftpad = itembox->get_left_pad (itembox);
  int item_rightpad = itembox->get_right_pad (itembox);
  int item_border_width = item_leftpad + item_rightpad;
  int item_toppad = itembox->get_top_pad (itembox);
342
  int sel_leftpad = selbox->get_left_pad (selbox);
343 344
  int sel_rightpad = selbox->get_right_pad (selbox);
  int sel_border_width = sel_leftpad + sel_rightpad;
345
  int sel_toppad = selbox->get_top_pad (selbox);
346 347 348 349

  int max_leftpad = grub_max (item_leftpad, sel_leftpad);
  int max_toppad = grub_max (item_toppad, sel_toppad);
  int item_top = 0;
Colin D Bennett's avatar
Colin D Bennett committed
350 351
  int menu_index;
  int visible_index;
352 353 354 355 356 357 358 359
  struct grub_video_rect oviewport;

  grub_video_get_viewport (&oviewport.x, &oviewport.y,
			   &oviewport.width, &oviewport.height);
  grub_video_set_viewport (oviewport.x + boxpad, 
			   oviewport.y + boxpad,
			   oviewport.width - 2 * boxpad,
			   oviewport.height - 2 * boxpad);
Colin D Bennett's avatar
Colin D Bennett committed
360

361
  int cwidth = oviewport.width - 2 * boxpad;
362 363 364 365 366 367 368 369 370 371

  itembox->set_content_size (itembox, cwidth - item_border_width,
                             text_box_height);
  selbox->set_content_size (selbox, cwidth - sel_border_width,
                            text_box_height);

  int text_left_offset = self->icon_width + icon_text_space;
  int item_text_top_offset = (text_box_height - (ascent + descent)) / 2 + ascent;
  int sel_text_top_offset = (text_box_height - (selected_ascent
                                                + selected_descent)) / 2
372
                                 + selected_ascent;
373 374

  grub_video_rect_t svpsave, sviewport;
375 376 377
  sviewport.x = max_leftpad + text_left_offset;
  int text_viewport_width = cwidth - sviewport.x;
  sviewport.height = text_box_height;
378

379
  grub_video_color_t item_color;
380
  grub_video_color_t sel_color;
381
  item_color = grub_video_map_rgba_color (self->item_color);
382 383 384 385 386 387 388 389 390
  sel_color = grub_video_map_rgba_color (self->selected_item_color);

  int item_box_top_offset = max_toppad - item_toppad;
  int sel_box_top_offset = max_toppad - sel_toppad;
  int item_viewport_width = text_viewport_width - item_rightpad;
  int sel_viewport_width = text_viewport_width - sel_rightpad;
  int tmp_icon_top_offset = (text_box_height - self->icon_height) / 2;
  int item_icon_top_offset = item_toppad + tmp_icon_top_offset;
  int sel_icon_top_offset = sel_toppad + tmp_icon_top_offset;
391

Colin D Bennett's avatar
Colin D Bennett committed
392
  for (visible_index = 0, menu_index = self->first_shown_index;
393
       visible_index < num_shown_items && menu_index < self->view->menu->size;
Colin D Bennett's avatar
Colin D Bennett committed
394 395
       visible_index++, menu_index++)
    {
396
      int is_selected = (menu_index == self->view->selected);
397
      struct grub_video_bitmap *icon;
398 399
      grub_font_t font;
      grub_video_color_t color;
400 401 402 403
      int text_top_offset;
      int top_pad;
      int icon_top_offset;
      int viewport_width;
Colin D Bennett's avatar
Colin D Bennett committed
404 405 406

      if (is_selected)
        {
407
          selbox->draw (selbox, 0, item_top + sel_box_top_offset);
408
          font = self->selected_item_font;
409 410 411 412 413
          color = sel_color;
          text_top_offset = sel_text_top_offset;
          top_pad = sel_toppad;
          icon_top_offset = sel_icon_top_offset;
          viewport_width = sel_viewport_width;
414 415 416
        }
      else
        {
417
          itembox->draw (itembox, 0, item_top + item_box_top_offset);
418 419
          font = self->item_font;
          color = item_color;
420 421 422 423
          text_top_offset = item_text_top_offset;
          top_pad = item_toppad;
          icon_top_offset = item_icon_top_offset;
          viewport_width = item_viewport_width;
Colin D Bennett's avatar
Colin D Bennett committed
424 425
        }

426 427
      icon = get_item_icon (self, menu_index);
      if (icon != 0)
Colin D Bennett's avatar
Colin D Bennett committed
428
        grub_video_blit_bitmap (icon, GRUB_VIDEO_BLIT_BLEND,
429 430
                                max_leftpad,
                                item_top + icon_top_offset,
Colin D Bennett's avatar
Colin D Bennett committed
431 432 433
                                0, 0, self->icon_width, self->icon_height);

      const char *item_title =
434
        grub_menu_get_entry (self->view->menu, menu_index)->title;
435

436 437
      sviewport.y = item_top + top_pad;
      sviewport.width = viewport_width;
438
      grub_gui_set_viewport (&sviewport, &svpsave);
Colin D Bennett's avatar
Colin D Bennett committed
439 440
      grub_font_draw_string (item_title,
                             font,
441
                             color,
442
                             0,
443
                             text_top_offset);
444
      grub_gui_restore_viewport (&svpsave);
Colin D Bennett's avatar
Colin D Bennett committed
445

446
      item_top += text_box_height + item_vspace;
Colin D Bennett's avatar
Colin D Bennett committed
447
    }
448 449 450 451
  grub_video_set_viewport (oviewport.x,
			   oviewport.y,
			   oviewport.width,
			   oviewport.height);
Colin D Bennett's avatar
Colin D Bennett committed
452 453 454
}

static void
455
list_paint (void *vself, const grub_video_rect_t *region)
Colin D Bennett's avatar
Colin D Bennett committed
456 457
{
  list_impl_t self = vself;
458
  grub_video_rect_t vpsave;
Colin D Bennett's avatar
Colin D Bennett committed
459 460 461

  if (! self->visible)
    return;
462 463
  if (!grub_video_have_common_points (region, &self->bounds))
    return;
Colin D Bennett's avatar
Colin D Bennett committed
464 465 466

  check_boxes (self);

467
  if (! self->menu_box || ! self->selected_item_box || ! self->item_box)
468 469
    return;

Colin D Bennett's avatar
Colin D Bennett committed
470
  grub_gui_set_viewport (&self->bounds, &vpsave);
471 472 473 474 475 476 477 478 479
  {
    grub_gfxmenu_box_t box = self->menu_box;
    int box_left_pad = box->get_left_pad (box);
    int box_top_pad = box->get_top_pad (box);
    int box_right_pad = box->get_right_pad (box);
    int box_bottom_pad = box->get_bottom_pad (box);
    grub_video_rect_t vpsave2, content_rect;
    int num_shown_items = get_num_shown_items (self);
    int drawing_scrollbar = (self->draw_scrollbar
480
			     && (num_shown_items < self->view->menu->size)
481
			     && check_scrollbar (self));
482
    int scrollbar_width = self->scrollbar_width;
483 484 485 486 487 488 489 490 491 492

    content_rect.x = box_left_pad;
    content_rect.y = box_top_pad;
    content_rect.width = self->bounds.width - box_left_pad - box_right_pad;
    content_rect.height = self->bounds.height - box_top_pad - box_bottom_pad;

    box->set_content_size (box, content_rect.width, content_rect.height);

    box->draw (box, 0, 0);

493 494 495
    switch (self->scrollbar_slice)
      {
        case SCROLLBAR_SLICE_WEST:
496 497
          content_rect.x += self->scrollbar_right_pad;
          content_rect.width -= self->scrollbar_right_pad;
498 499 500
          break;
        case SCROLLBAR_SLICE_CENTER:
          if (drawing_scrollbar)
501 502
            content_rect.width -= scrollbar_width + self->scrollbar_left_pad
                                  + self->scrollbar_right_pad;
503 504
          break;
        case SCROLLBAR_SLICE_EAST:
505
          content_rect.width -= self->scrollbar_left_pad;
506 507 508
          break;
      }

509
    grub_gui_set_viewport (&content_rect, &vpsave2);
510
    draw_menu (self, num_shown_items);
511 512 513
    grub_gui_restore_viewport (&vpsave2);

    if (drawing_scrollbar)
514
      {
515 516 517
        content_rect.y += self->scrollbar_top_pad;
        content_rect.height -= self->scrollbar_top_pad
                               + self->scrollbar_bottom_pad;
518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534
        content_rect.width = scrollbar_width;
        switch (self->scrollbar_slice)
          {
            case SCROLLBAR_SLICE_WEST:
              if (box_left_pad > scrollbar_width)
                {
                  content_rect.x = box_left_pad - scrollbar_width;
                  content_rect.width = scrollbar_width;
                }
              else
                {
                  content_rect.x = 0;
                  content_rect.width = box_left_pad;
                }
              break;
            case SCROLLBAR_SLICE_CENTER:
              content_rect.x = self->bounds.width - box_right_pad
535
                               - scrollbar_width - self->scrollbar_right_pad;
536 537 538 539 540 541 542
              content_rect.width = scrollbar_width;
              break;
            case SCROLLBAR_SLICE_EAST:
              content_rect.x = self->bounds.width - box_right_pad;
              content_rect.width = box_right_pad;
              break;
          }
543 544 545 546 547

        grub_gui_set_viewport (&content_rect, &vpsave2);
        draw_scrollbar (self,
                        self->first_shown_index, num_shown_items,
                        0, self->view->menu->size,
548
                        scrollbar_width,
549 550 551
                        content_rect.height);
        grub_gui_restore_viewport (&vpsave2);
      }
552 553
  }

Colin D Bennett's avatar
Colin D Bennett committed
554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585
  grub_gui_restore_viewport (&vpsave);
}

static void
list_set_parent (void *vself, grub_gui_container_t parent)
{
  list_impl_t self = vself;
  self->parent = parent;
}

static grub_gui_container_t
list_get_parent (void *vself)
{
  list_impl_t self = vself;
  return self->parent;
}

static void
list_set_bounds (void *vself, const grub_video_rect_t *bounds)
{
  list_impl_t self = vself;
  self->bounds = *bounds;
}

static void
list_get_bounds (void *vself, grub_video_rect_t *bounds)
{
  list_impl_t self = vself;
  *bounds = self->bounds;
}

static void
586
list_get_minimal_size (void *vself, unsigned *width, unsigned *height)
Colin D Bennett's avatar
Colin D Bennett committed
587 588 589 590 591 592 593 594
{
  list_impl_t self = vself;

  if (check_boxes (self))
    {
      int boxpad = self->item_padding;
      int item_vspace = self->item_spacing;
      int item_height = self->item_height;
595
      int num_items = 3;
Colin D Bennett's avatar
Colin D Bennett committed
596 597 598 599 600 601

      grub_gfxmenu_box_t box = self->menu_box;
      int box_left_pad = box->get_left_pad (box);
      int box_top_pad = box->get_top_pad (box);
      int box_right_pad = box->get_right_pad (box);
      int box_bottom_pad = box->get_bottom_pad (box);
602
      unsigned width_s;
603 604

      grub_gfxmenu_box_t selbox = self->selected_item_box;
605 606
      int sel_top_pad = selbox->get_top_pad (selbox);
      int sel_bottom_pad = selbox->get_bottom_pad (selbox);
607 608 609
      int sel_left_pad = selbox->get_left_pad (selbox);
      int sel_right_pad = selbox->get_right_pad (selbox);

610 611 612 613 614 615 616 617 618 619 620
      grub_gfxmenu_box_t itembox = self->item_box;
      int item_top_pad = itembox->get_top_pad (itembox);
      int item_bottom_pad = itembox->get_bottom_pad (itembox);
      int item_left_pad = itembox->get_left_pad (itembox);
      int item_right_pad = itembox->get_right_pad (itembox);

      int max_left_pad = grub_max (item_left_pad, sel_left_pad);
      int max_right_pad = grub_max (item_right_pad, sel_right_pad);
      int max_top_pad = grub_max (item_top_pad, sel_top_pad);
      int max_bottom_pad = grub_max (item_bottom_pad, sel_bottom_pad);

621 622 623 624 625
      *width = grub_font_get_string_width (self->item_font, "Typical OS");
      width_s = grub_font_get_string_width (self->selected_item_font,
					    "Typical OS");
      if (*width < width_s)
	*width = width_s;
Colin D Bennett's avatar
Colin D Bennett committed
626

627
      *width += 2 * boxpad + box_left_pad + box_right_pad
628
                + max_left_pad + max_right_pad
629
                + self->item_icon_space + self->icon_width;
Colin D Bennett's avatar
Colin D Bennett committed
630

631 632 633
      switch (self->scrollbar_slice)
        {
          case SCROLLBAR_SLICE_WEST:
634
            *width += self->scrollbar_right_pad;
635 636
            break;
          case SCROLLBAR_SLICE_CENTER:
637 638
            *width += self->scrollbar_width + self->scrollbar_left_pad
                      + self->scrollbar_right_pad;
639 640
            break;
          case SCROLLBAR_SLICE_EAST:
641
            *width += self->scrollbar_left_pad;
642 643 644
            break;
        }

Colin D Bennett's avatar
Colin D Bennett committed
645 646 647 648
      /* Set the menu box height to fit the items.  */
      *height = (item_height * num_items
                 + item_vspace * (num_items - 1)
                 + 2 * boxpad
649
                 + box_top_pad + box_bottom_pad
650
                 + max_top_pad + max_bottom_pad);
Colin D Bennett's avatar
Colin D Bennett committed
651 652 653 654 655 656 657 658 659 660 661 662 663 664 665
    }
  else
    {
      *width = 0;
      *height = 0;
    }
}

static grub_err_t
list_set_property (void *vself, const char *name, const char *value)
{
  list_impl_t self = vself;
  if (grub_strcmp (name, "item_font") == 0)
    {
      self->item_font = grub_font_get (value);
666 667
      if (self->selected_item_font_inherit)
        self->selected_item_font = self->item_font;
Colin D Bennett's avatar
Colin D Bennett committed
668 669 670 671
    }
  else if (grub_strcmp (name, "selected_item_font") == 0)
    {
      if (! value || grub_strcmp (value, "inherit") == 0)
672 673 674 675
        {
          self->selected_item_font = self->item_font;
          self->selected_item_font_inherit = 1;
        }
Colin D Bennett's avatar
Colin D Bennett committed
676
      else
677 678 679 680
        {
          self->selected_item_font = grub_font_get (value);
          self->selected_item_font_inherit = 0;
        }
Colin D Bennett's avatar
Colin D Bennett committed
681 682 683
    }
  else if (grub_strcmp (name, "item_color") == 0)
    {
684 685 686 687 688 689 690
      grub_video_rgba_color_t color;
      if (grub_video_parse_color (value, &color) == GRUB_ERR_NONE)
        {
          self->item_color = color;
          if (self->selected_item_color_inherit)
            self->selected_item_color = self->item_color;
        }
Colin D Bennett's avatar
Colin D Bennett committed
691 692 693 694 695
    }
  else if (grub_strcmp (name, "selected_item_color") == 0)
    {
      if (! value || grub_strcmp (value, "inherit") == 0)
        {
696 697
          self->selected_item_color = self->item_color;
          self->selected_item_color_inherit = 1;
Colin D Bennett's avatar
Colin D Bennett committed
698 699 700
        }
      else
        {
701 702
          grub_video_rgba_color_t color;
          if (grub_video_parse_color (value, &color)
Colin D Bennett's avatar
Colin D Bennett committed
703
              == GRUB_ERR_NONE)
704 705 706 707
            {
              self->selected_item_color = color;
              self->selected_item_color_inherit = 0;
            }
Colin D Bennett's avatar
Colin D Bennett committed
708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749
        }
    }
  else if (grub_strcmp (name, "icon_width") == 0)
    {
      self->icon_width = grub_strtol (value, 0, 10);
      grub_gfxmenu_icon_manager_set_icon_size (self->icon_manager,
                                               self->icon_width,
                                               self->icon_height);
    }
  else if (grub_strcmp (name, "icon_height") == 0)
    {
      self->icon_height = grub_strtol (value, 0, 10);
      grub_gfxmenu_icon_manager_set_icon_size (self->icon_manager,
                                               self->icon_width,
                                               self->icon_height);
    }
  else if (grub_strcmp (name, "item_height") == 0)
    {
      self->item_height = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "item_padding") == 0)
    {
      self->item_padding = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "item_icon_space") == 0)
    {
      self->item_icon_space = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "item_spacing") == 0)
    {
      self->item_spacing = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "visible") == 0)
    {
      self->visible = grub_strcmp (value, "false") != 0;
    }
  else if (grub_strcmp (name, "menu_pixmap_style") == 0)
    {
      self->need_to_recreate_boxes = 1;
      grub_free (self->menu_box_pattern);
      self->menu_box_pattern = value ? grub_strdup (value) : 0;
    }
750
  else if (grub_strcmp (name, "item_pixmap_style") == 0)
Colin D Bennett's avatar
Colin D Bennett committed
751 752
    {
      self->need_to_recreate_boxes = 1;
753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776
      grub_free (self->item_box_pattern);
      self->item_box_pattern = value ? grub_strdup (value) : 0;
      if (self->selected_item_box_pattern_inherit)
        {
          grub_free (self->selected_item_box_pattern);
          self->selected_item_box_pattern = value ? grub_strdup (value) : 0;
        }
    }
  else if (grub_strcmp (name, "selected_item_pixmap_style") == 0)
    {
      if (!value || grub_strcmp (value, "inherit") == 0)
        {
          grub_free (self->selected_item_box_pattern);
          char *tmp = self->item_box_pattern;
          self->selected_item_box_pattern = tmp ? grub_strdup (tmp) : 0;
          self->selected_item_box_pattern_inherit = 1;
        }
      else
        {
          self->need_to_recreate_boxes = 1;
          grub_free (self->selected_item_box_pattern);
          self->selected_item_box_pattern = value ? grub_strdup (value) : 0;
          self->selected_item_box_pattern_inherit = 0;
        }
Colin D Bennett's avatar
Colin D Bennett committed
777 778 779 780 781 782 783 784 785 786 787 788 789
    }
  else if (grub_strcmp (name, "scrollbar_frame") == 0)
    {
      self->need_to_recreate_scrollbar = 1;
      grub_free (self->scrollbar_frame_pattern);
      self->scrollbar_frame_pattern = value ? grub_strdup (value) : 0;
    }
  else if (grub_strcmp (name, "scrollbar_thumb") == 0)
    {
      self->need_to_recreate_scrollbar = 1;
      grub_free (self->scrollbar_thumb_pattern);
      self->scrollbar_thumb_pattern = value ? grub_strdup (value) : 0;
    }
790 791 792 793
  else if (grub_strcmp (name, "scrollbar_thumb_overlay") == 0)
    {
      self->scrollbar_thumb_overlay = grub_strcmp (value, "true") == 0;
    }
Colin D Bennett's avatar
Colin D Bennett committed
794 795 796 797
  else if (grub_strcmp (name, "scrollbar_width") == 0)
    {
      self->scrollbar_width = grub_strtol (value, 0, 10);
    }
798 799 800 801 802 803 804 805 806
  else if (grub_strcmp (name, "scrollbar_slice") == 0)
    {
      if (grub_strcmp (value, "west") == 0)
        self->scrollbar_slice = SCROLLBAR_SLICE_WEST;
      else if (grub_strcmp (value, "center") == 0)
        self->scrollbar_slice = SCROLLBAR_SLICE_CENTER;
      else if (grub_strcmp (value, "east") == 0)
        self->scrollbar_slice = SCROLLBAR_SLICE_EAST;
    }
807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822
  else if (grub_strcmp (name, "scrollbar_left_pad") == 0)
    {
      self->scrollbar_left_pad = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "scrollbar_right_pad") == 0)
    {
      self->scrollbar_right_pad = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "scrollbar_top_pad") == 0)
    {
      self->scrollbar_top_pad = grub_strtol (value, 0, 10);
    }
  else if (grub_strcmp (name, "scrollbar_bottom_pad") == 0)
    {
      self->scrollbar_bottom_pad = grub_strtol (value, 0, 10);
    }
Colin D Bennett's avatar
Colin D Bennett committed
823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846
  else if (grub_strcmp (name, "scrollbar") == 0)
    {
      self->draw_scrollbar = grub_strcmp (value, "false") != 0;
    }
  else if (grub_strcmp (name, "theme_dir") == 0)
    {
      self->need_to_recreate_boxes = 1;
      grub_free (self->theme_dir);
      self->theme_dir = value ? grub_strdup (value) : 0;
    }
  else if (grub_strcmp (name, "id") == 0)
    {
      grub_free (self->id);
      if (value)
        self->id = grub_strdup (value);
      else
        self->id = 0;
    }
  return grub_errno;
}

/* Set necessary information that the gfxmenu view provides.  */
static void
list_set_view_info (void *vself,
847
                    grub_gfxmenu_view_t view)
Colin D Bennett's avatar
Colin D Bennett committed
848 849
{
  list_impl_t self = vself;
850 851 852
  grub_gfxmenu_icon_manager_set_theme_path (self->icon_manager,
					    view->theme_path);
  self->view = view;
Colin D Bennett's avatar
Colin D Bennett committed
853 854
}

855 856 857 858 859 860 861 862 863 864
/* Refresh list variables */
static void
list_refresh_info (void *vself,
                   grub_gfxmenu_view_t view)
{
  list_impl_t self = vself;
  if (view->nested)
    self->first_shown_index = 0;
}

865 866 867 868 869 870 871 872 873 874 875 876 877 878
static struct grub_gui_component_ops list_comp_ops =
  {
    .destroy = list_destroy,
    .get_id = list_get_id,
    .is_instance = list_is_instance,
    .paint = list_paint,
    .set_parent = list_set_parent,
    .get_parent = list_get_parent,
    .set_bounds = list_set_bounds,
    .get_bounds = list_get_bounds,
    .get_minimal_size = list_get_minimal_size,
    .set_property = list_set_property
  };

Colin D Bennett's avatar
Colin D Bennett committed
879 880
static struct grub_gui_list_ops list_ops =
{
881 882
  .set_view_info = list_set_view_info,
  .refresh_list = list_refresh_info
Colin D Bennett's avatar
Colin D Bennett committed
883 884 885 886 887 888 889
};

grub_gui_component_t
grub_gui_list_new (void)
{
  list_impl_t self;
  grub_font_t default_font;
890
  grub_video_rgba_color_t default_fg_color;
Colin D Bennett's avatar
Colin D Bennett committed
891

892
  self = grub_zalloc (sizeof (*self));
Colin D Bennett's avatar
Colin D Bennett committed
893 894 895
  if (! self)
    return 0;

896 897 898
  self->list.ops = &list_ops;
  self->list.component.ops = &list_comp_ops;

Colin D Bennett's avatar
Colin D Bennett committed
899 900
  self->visible = 1;

901
  default_font = grub_font_get ("Unknown Regular 16");
902
  default_fg_color = grub_video_rgba_color_rgb (0, 0, 0);
Colin D Bennett's avatar
Colin D Bennett committed
903 904 905 906 907 908 909 910

  self->icon_width = 32;
  self->icon_height = 32;
  self->item_height = 42;
  self->item_padding = 14;
  self->item_icon_space = 4;
  self->item_spacing = 16;
  self->item_font = default_font;
911 912
  self->selected_item_font_inherit = 1; /* Default to using the item_font.  */
  self->selected_item_font = default_font;
Colin D Bennett's avatar
Colin D Bennett committed
913
  self->item_color = default_fg_color;
914
  self->selected_item_color_inherit = 1;  /* Default to using the item_color.  */
Colin D Bennett's avatar
Colin D Bennett committed
915 916 917 918 919 920 921 922
  self->selected_item_color = default_fg_color;

  self->draw_scrollbar = 1;
  self->need_to_recreate_scrollbar = 1;
  self->scrollbar_frame = 0;
  self->scrollbar_thumb = 0;
  self->scrollbar_frame_pattern = 0;
  self->scrollbar_thumb_pattern = 0;
923
  self->scrollbar_thumb_overlay = 0;
Colin D Bennett's avatar
Colin D Bennett committed
924
  self->scrollbar_width = 16;
925
  self->scrollbar_slice = SCROLLBAR_SLICE_EAST;
926 927 928 929
  self->scrollbar_left_pad = 2;
  self->scrollbar_right_pad = 0;
  self->scrollbar_top_pad = 0;
  self->scrollbar_bottom_pad = 0;
Colin D Bennett's avatar
Colin D Bennett committed
930 931 932 933 934 935

  self->first_shown_index = 0;

  self->need_to_recreate_boxes = 0;
  self->theme_dir = 0;
  self->menu_box_pattern = 0;
936 937
  self->item_box_pattern = 0;
  self->selected_item_box_pattern_inherit = 1;/*Default to using the item_box.*/
Colin D Bennett's avatar
Colin D Bennett committed
938 939
  self->selected_item_box_pattern = 0;
  self->menu_box = grub_gfxmenu_create_box (0, 0);
940
  self->item_box = grub_gfxmenu_create_box (0, 0);
Colin D Bennett's avatar
Colin D Bennett committed
941 942 943 944 945
  self->selected_item_box = grub_gfxmenu_create_box (0, 0);

  self->icon_manager = grub_gfxmenu_icon_manager_new ();
  if (! self->icon_manager)
    {
946
      self->list.component.ops->destroy (self);
Colin D Bennett's avatar
Colin D Bennett committed
947 948 949 950 951 952 953
      return 0;
    }
  grub_gfxmenu_icon_manager_set_icon_size (self->icon_manager,
                                           self->icon_width,
                                           self->icon_height);
  return (grub_gui_component_t) self;
}