From now on, libmenu does not steal all input keys from input modules.
[mplayer/greg.git] / libmenu / menu.c
blobdd9d6d8ec918a1d27820c28e7bee594f834ca297
2 #include "config.h"
3 #include "mp_msg.h"
4 #include "help_mp.h"
6 #include <stdlib.h>
7 #include <stdio.h>
8 #include <string.h>
9 #include <fcntl.h>
10 #include <unistd.h>
12 #include "libvo/osd.h"
13 #include "libvo/font_load.h"
14 #include "libvo/sub.h"
15 #include "osdep/keycodes.h"
16 #include "asxparser.h"
17 #include "stream/stream.h"
18 #include "input/input.h"
20 #include "libmpcodecs/img_format.h"
21 #include "libmpcodecs/mp_image.h"
22 #include "m_option.h"
23 #include "m_struct.h"
24 #include "menu.h"
26 extern menu_info_t menu_info_cmdlist;
27 extern menu_info_t menu_info_chapsel;
28 extern menu_info_t menu_info_pt;
29 extern menu_info_t menu_info_filesel;
30 extern menu_info_t menu_info_txt;
31 extern menu_info_t menu_info_console;
32 extern menu_info_t menu_info_pref;
33 #ifdef HAS_DVBIN_SUPPORT
34 extern menu_info_t menu_info_dvbsel;
35 #endif
38 menu_info_t* menu_info_list[] = {
39 &menu_info_pt,
40 &menu_info_cmdlist,
41 &menu_info_chapsel,
42 &menu_info_filesel,
43 &menu_info_txt,
44 &menu_info_console,
45 #ifdef HAS_DVBIN_SUPPORT
46 &menu_info_dvbsel,
47 #endif
48 &menu_info_pref,
49 NULL
52 typedef struct key_cmd_s {
53 int key;
54 char *cmd;
55 } key_cmd_t;
57 typedef struct menu_cmd_bindings_s {
58 char *name;
59 key_cmd_t *bindings;
60 int binding_num;
61 struct menu_cmd_bindings_s *parent;
62 } menu_cmd_bindings_t;
64 struct menu_def_st {
65 char* name;
66 menu_info_t* type;
67 void* cfg;
68 char* args;
71 double menu_mouse_x = -1.0;
72 double menu_mouse_y = -1.0;
73 int menu_mouse_pos_updated = 0;
75 static struct MPContext *menu_ctx = NULL;
76 static menu_def_t* menu_list = NULL;
77 static int menu_count = 0;
78 static menu_cmd_bindings_t *cmd_bindings = NULL;
79 static int cmd_bindings_num = 0;
82 menu_cmd_bindings_t *get_cmd_bindings(const char *name) {
83 int i;
84 for (i = 0; i < cmd_bindings_num; ++i)
85 if (!strcasecmp(cmd_bindings[i].name, name))
86 return &cmd_bindings[i];
87 return NULL;
90 static int menu_parse_config(char* buffer) {
91 char *element,*body, **attribs, *name;
92 menu_info_t* minfo = NULL;
93 int r,i;
94 ASX_Parser_t* parser = asx_parser_new();
96 while(1) {
97 r = asx_get_element(parser,&buffer,&element,&body,&attribs);
98 if(r < 0) {
99 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,parser->line);
100 asx_parser_free(parser);
101 return 0;
102 } else if(r == 0) {
103 asx_parser_free(parser);
104 return 1;
106 // Has it a name ?
107 name = asx_get_attrib("name",attribs);
108 if(!name) {
109 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuDefinitionsNeedANameAttrib,parser->line);
110 free(element);
111 if(body) free(body);
112 asx_free_attribs(attribs);
113 continue;
116 if (!strcasecmp(element, "keybindings")) {
117 menu_cmd_bindings_t *bindings = cmd_bindings;
118 char *parent_bindings;
119 cmd_bindings = realloc(cmd_bindings,
120 (cmd_bindings_num+1)*sizeof(menu_cmd_bindings_t));
121 for (i = 0; i < cmd_bindings_num; ++i)
122 if (cmd_bindings[i].parent)
123 cmd_bindings[i].parent = cmd_bindings[i].parent-bindings+cmd_bindings;
124 bindings = &cmd_bindings[cmd_bindings_num];
125 memset(bindings, 0, sizeof(menu_cmd_bindings_t));
126 bindings->name = name;
127 parent_bindings = asx_get_attrib("parent",attribs);
128 if (parent_bindings) {
129 bindings->parent = get_cmd_bindings(parent_bindings);
130 free(parent_bindings);
132 free(element);
133 asx_free_attribs(attribs);
134 if (body) {
135 char *bd = body;
136 char *b, *key, *cmd;
137 int keycode;
138 for(;;) {
139 r = asx_get_element(parser,&bd,&element,&b,&attribs);
140 if(r < 0) {
141 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,
142 parser->line);
143 free(body);
144 asx_parser_free(parser);
145 return 0;
147 if(r == 0)
148 break;
149 if (!strcasecmp(element, "binding")) {
150 key = asx_get_attrib("key",attribs);
151 cmd = asx_get_attrib("cmd",attribs);
152 if (key && (keycode = mp_input_get_key_from_name(key)) >= 0) {
153 keycode &= ~MP_NO_REPEAT_KEY;
154 mp_msg(MSGT_GLOBAL,MSGL_V,
155 "[libmenu] got keybinding element %d %s=>[%s].\n",
156 keycode, key, cmd ? cmd : "");
157 bindings->bindings = realloc(bindings->bindings,
158 (bindings->binding_num+1)*sizeof(key_cmd_t));
159 bindings->bindings[bindings->binding_num].key = keycode;
160 bindings->bindings[bindings->binding_num].cmd = cmd;
161 ++bindings->binding_num;
163 else
164 free(cmd);
165 free(key);
167 free(element);
168 asx_free_attribs(attribs);
169 free(b);
171 free(body);
173 ++cmd_bindings_num;
174 continue;
176 // Try to find this menu type in our list
177 for(i = 0, minfo = NULL ; menu_info_list[i] ; i++) {
178 if(strcasecmp(element,menu_info_list[i]->name) == 0) {
179 minfo = menu_info_list[i];
180 break;
183 // Got it : add this to our list
184 if(minfo) {
185 menu_list = realloc(menu_list,(menu_count+2)*sizeof(menu_def_t));
186 menu_list[menu_count].name = name;
187 menu_list[menu_count].type = minfo;
188 menu_list[menu_count].cfg = m_struct_alloc(&minfo->priv_st);
189 menu_list[menu_count].args = body;
190 // Setup the attribs
191 for(i = 0 ; attribs[2*i] ; i++) {
192 if(strcasecmp(attribs[2*i],"name") == 0) continue;
193 if(!m_struct_set(&minfo->priv_st,menu_list[menu_count].cfg,attribs[2*i], attribs[2*i+1]))
194 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_BadAttrib,attribs[2*i],attribs[2*i+1],
195 name,parser->line);
197 menu_count++;
198 memset(&menu_list[menu_count],0,sizeof(menu_def_t));
199 } else {
200 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnknownMenuType,element,parser->line);
201 free(name);
202 if(body) free(body);
205 free(element);
206 asx_free_attribs(attribs);
212 /// This will build the menu_defs list from the cfg file
213 #define BUF_STEP 1024
214 #define BUF_MIN 128
215 #define BUF_MAX BUF_STEP*1024
216 int menu_init(struct MPContext *mpctx, char* cfg_file) {
217 char* buffer = NULL;
218 int bl = BUF_STEP, br = 0;
219 int f, fd;
220 #ifndef HAVE_FREETYPE
221 if(vo_font == NULL)
222 return 0;
223 #endif
224 fd = open(cfg_file, O_RDONLY);
225 if(fd < 0) {
226 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_CantOpenConfigFile,cfg_file);
227 return 0;
229 buffer = malloc(bl);
230 while(1) {
231 int r;
232 if(bl - br < BUF_MIN) {
233 if(bl >= BUF_MAX) {
234 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsTooBig,BUF_MAX/1024);
235 close(fd);
236 free(buffer);
237 return 0;
239 bl += BUF_STEP;
240 buffer = realloc(buffer,bl);
242 r = read(fd,buffer+br,bl-br);
243 if(r == 0) break;
244 br += r;
246 if(!br) {
247 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsEmpty);
248 return 0;
250 buffer[br-1] = '\0';
252 close(fd);
254 menu_ctx = mpctx;
255 f = menu_parse_config(buffer);
256 free(buffer);
257 return f;
260 // Destroy all this stuff
261 void menu_uninit(void) {
262 int i;
263 for(i = 0 ; menu_list && menu_list[i].name ; i++) {
264 free(menu_list[i].name);
265 m_struct_free(&menu_list[i].type->priv_st,menu_list[i].cfg);
266 if(menu_list[i].args) free(menu_list[i].args);
268 free(menu_list);
269 menu_count = 0;
270 for (i = 0; i < cmd_bindings_num; ++i) {
271 free(cmd_bindings[i].name);
272 while(cmd_bindings[i].binding_num > 0)
273 free(cmd_bindings[i].bindings[--cmd_bindings[i].binding_num].cmd);
274 free(cmd_bindings[i].bindings);
276 free(cmd_bindings);
279 /// Default read_key function
280 int menu_dflt_read_key(menu_t* menu,int cmd) {
281 int i;
282 menu_cmd_bindings_t *bindings = get_cmd_bindings(menu->type->name);
283 if (!bindings)
284 bindings = get_cmd_bindings(menu->type->type->name);
285 if (!bindings)
286 bindings = get_cmd_bindings("default");
287 while (bindings) {
288 for (i = 0; i < bindings->binding_num; ++i) {
289 if (bindings->bindings[i].key == cmd) {
290 if (bindings->bindings[i].cmd)
291 mp_input_parse_and_queue_cmds(bindings->bindings[i].cmd);
292 return 1;
295 bindings = bindings->parent;
297 return 0;
300 menu_t* menu_open(char *name) {
301 menu_t* m;
302 int i;
304 for(i = 0 ; menu_list[i].name != NULL ; i++) {
305 if(strcmp(name,menu_list[i].name) == 0)
306 break;
308 if(menu_list[i].name == NULL) {
309 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuNotFound,name);
310 return NULL;
312 m = calloc(1,sizeof(menu_t));
313 m->priv_st = &(menu_list[i].type->priv_st);
314 m->priv = m_struct_copy(m->priv_st,menu_list[i].cfg);
315 m->ctx = menu_ctx;
316 m->type = &menu_list[i];
317 if(menu_list[i].type->open(m,menu_list[i].args))
318 return m;
319 if(m->priv)
320 m_struct_free(m->priv_st,m->priv);
321 free(m);
322 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuInitFailed,name);
323 return NULL;
326 void menu_draw(menu_t* menu,mp_image_t* mpi) {
327 if(menu->show && menu->draw)
328 menu->draw(menu,mpi);
331 void menu_update_mouse_pos(double x, double y) {
332 menu_mouse_x = x;
333 menu_mouse_y = y;
334 menu_mouse_pos_updated = 1;
337 void menu_read_cmd(menu_t* menu,int cmd) {
338 if(menu->read_cmd)
339 menu->read_cmd(menu,cmd);
342 void menu_close(menu_t* menu) {
343 if(menu->close)
344 menu->close(menu);
345 if(menu->priv)
346 m_struct_free(menu->priv_st,menu->priv);
347 free(menu);
350 int menu_read_key(menu_t* menu,int cmd) {
351 if(menu->read_key)
352 return menu->read_key(menu,cmd);
353 else
354 return menu_dflt_read_key(menu,cmd);
357 ///////////////////////////// Helpers ////////////////////////////////////
359 typedef void (*draw_alpha_f)(int w,int h, unsigned char* src, unsigned char *srca, int srcstride, unsigned char* dstbase,int dststride);
361 inline static draw_alpha_f get_draw_alpha(uint32_t fmt) {
362 switch(fmt) {
363 case IMGFMT_BGR15:
364 case IMGFMT_RGB15:
365 return vo_draw_alpha_rgb15;
366 case IMGFMT_BGR16:
367 case IMGFMT_RGB16:
368 return vo_draw_alpha_rgb16;
369 case IMGFMT_BGR24:
370 case IMGFMT_RGB24:
371 return vo_draw_alpha_rgb24;
372 case IMGFMT_BGR32:
373 case IMGFMT_RGB32:
374 return vo_draw_alpha_rgb32;
375 case IMGFMT_YV12:
376 case IMGFMT_I420:
377 case IMGFMT_IYUV:
378 case IMGFMT_YVU9:
379 case IMGFMT_IF09:
380 case IMGFMT_Y800:
381 case IMGFMT_Y8:
382 return vo_draw_alpha_yv12;
383 case IMGFMT_YUY2:
384 return vo_draw_alpha_yuy2;
385 case IMGFMT_UYVY:
386 return vo_draw_alpha_uyvy;
389 return NULL;
392 // return the real height of a char:
393 static inline int get_height(int c,int h){
394 int font;
395 if ((font=vo_font->font[c])>=0)
396 if(h<vo_font->pic_a[font]->h) h=vo_font->pic_a[font]->h;
397 return h;
400 static void render_txt(char *txt)
402 while (*txt) {
403 int c = utf8_get_char((const char**)&txt);
404 render_one_glyph(vo_font, c);
408 #ifdef USE_FRIBIDI
409 #include <fribidi/fribidi.h>
410 #include "libavutil/common.h"
411 char *menu_fribidi_charset = NULL;
412 int menu_flip_hebrew = 0;
413 int menu_fribidi_flip_commas = 0;
415 static char *menu_fribidi(char *txt)
417 static int char_set_num = -1;
418 static FriBidiChar *logical, *visual;
419 static size_t buffer_size = 1024;
420 static char *outputstr;
422 FriBidiCharType base;
423 fribidi_boolean log2vis;
424 size_t len;
426 if (menu_flip_hebrew) {
427 len = strlen(txt);
428 if (char_set_num == -1) {
429 fribidi_set_mirroring (1);
430 fribidi_set_reorder_nsm (0);
431 char_set_num = fribidi_parse_charset("UTF-8");
432 buffer_size = FFMAX(1024,len+1);
433 logical = malloc(buffer_size);
434 visual = malloc(buffer_size);
435 outputstr = malloc(buffer_size);
436 } else if (len+1 > buffer_size) {
437 buffer_size = len+1;
438 logical = realloc(logical, buffer_size);
439 visual = realloc(visual, buffer_size);
440 outputstr = realloc(outputstr, buffer_size);
442 len = fribidi_charset_to_unicode (char_set_num, txt, len, logical);
443 base = menu_fribidi_flip_commas?FRIBIDI_TYPE_ON:FRIBIDI_TYPE_L;
444 log2vis = fribidi_log2vis (logical, len, &base, visual, NULL, NULL, NULL);
445 if (log2vis) {
446 len = fribidi_remove_bidi_marks (visual, len, NULL, NULL, NULL);
447 fribidi_unicode_to_charset (char_set_num, visual, len, outputstr);
448 return outputstr;
451 return txt;
453 #endif
455 void menu_draw_text(mp_image_t* mpi,char* txt, int x, int y) {
456 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
457 int font;
459 if(!draw_alpha) {
460 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
461 return;
464 #ifdef USE_FRIBIDI
465 txt = menu_fribidi(txt);
466 #endif
467 render_txt(txt);
469 while (*txt) {
470 int c=utf8_get_char((const char**)&txt);
471 if ((font=vo_font->font[c])>=0 && (x + vo_font->width[c] <= mpi->w) && (y + vo_font->pic_a[font]->h <= mpi->h))
472 draw_alpha(vo_font->width[c], vo_font->pic_a[font]->h,
473 vo_font->pic_b[font]->bmp+vo_font->start[c],
474 vo_font->pic_a[font]->bmp+vo_font->start[c],
475 vo_font->pic_a[font]->w,
476 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
477 mpi->stride[0]);
478 x+=vo_font->width[c]+vo_font->charspace;
483 void menu_draw_text_full(mp_image_t* mpi,char* txt,
484 int x, int y,int w, int h,
485 int vspace, int warp, int align, int anchor) {
486 int need_w,need_h;
487 int sy, ymin, ymax;
488 int sx, xmin, xmax, xmid, xrmin;
489 int ll = 0;
490 int font;
491 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
493 if(!draw_alpha) {
494 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
495 return;
498 #ifdef USE_FRIBIDI
499 txt = menu_fribidi(txt);
500 #endif
501 render_txt(txt);
503 if(x > mpi->w || y > mpi->h)
504 return;
506 if(anchor & MENU_TEXT_VCENTER) {
507 if(h <= 0) h = mpi->h;
508 ymin = y - h/2;
509 ymax = y + h/2;
510 } else if(anchor & MENU_TEXT_BOT) {
511 if(h <= 0) h = mpi->h - y;
512 ymin = y - h;
513 ymax = y;
514 } else {
515 if(h <= 0) h = mpi->h - y;
516 ymin = y;
517 ymax = y + h;
520 if(anchor & MENU_TEXT_HCENTER) {
521 if(w <= 0) w = mpi->w;
522 xmin = x - w/2;
523 xmax = x + w/2;
524 } else if(anchor & MENU_TEXT_RIGHT) {
525 if(w <= 0) w = mpi->w -x;
526 xmin = x - w;
527 xmax = x;
528 } else {
529 if(w <= 0) w = mpi->w -x;
530 xmin = x;
531 xmax = x + w;
534 // How many space do we need to draw this ?
535 menu_text_size(txt,w,vspace,warp,&need_w,&need_h);
537 // Find the first line
538 if(align & MENU_TEXT_VCENTER)
539 sy = ymin + ((h - need_h)/2);
540 else if(align & MENU_TEXT_BOT)
541 sy = ymax - need_h - 1;
542 else
543 sy = y;
545 #if 0
546 // Find the first col
547 if(align & MENU_TEXT_HCENTER)
548 sx = xmin + ((w - need_w)/2);
549 else if(align & MENU_TEXT_RIGHT)
550 sx = xmax - need_w;
551 #endif
553 xmid = xmin + (xmax - xmin) / 2;
554 xrmin = xmin;
555 // Clamp the bb to the mpi size
556 if(ymin < 0) ymin = 0;
557 if(xmin < 0) xmin = 0;
558 if(ymax > mpi->h) ymax = mpi->h;
559 if(xmax > mpi->w) xmax = mpi->w;
561 // Jump some the beginnig text if needed
562 while(sy < ymin && *txt) {
563 int c=utf8_get_char((const char**)&txt);
564 if(c == '\n' || (warp && ll + vo_font->width[c] > w)) {
565 ll = 0;
566 sy += vo_font->height + vspace;
567 if(c == '\n') continue;
569 ll += vo_font->width[c]+vo_font->charspace;
571 if(*txt == '\0') // Nothing left to draw
572 return;
574 while(sy < ymax && *txt) {
575 char* line_end = NULL;
576 int n;
578 if(txt[0] == '\n') { // New line
579 sy += vo_font->height + vspace;
580 txt++;
581 continue;
584 // Get the length and end of this line
585 for(n = 0, ll = 0 ; txt[n] != '\0' && txt[n] != '\n' ; n++) {
586 unsigned char c = txt[n];
587 if(warp && ll + vo_font->width[c] > w) break;
588 ll += vo_font->width[c]+vo_font->charspace;
590 line_end = &txt[n];
591 ll -= vo_font->charspace;
594 if(align & (MENU_TEXT_HCENTER|MENU_TEXT_RIGHT)) {
595 // Too long line
596 if(ll > xmax-xmin) {
597 if(align & MENU_TEXT_HCENTER) {
598 int mid = ll/2;
599 // Find the middle point
600 for(n--, ll = 0 ; n <= 0 ; n--) {
601 ll += vo_font->width[(int)txt[n]]+vo_font->charspace;
602 if(ll - vo_font->charspace > mid) break;
604 ll -= vo_font->charspace;
605 sx = xmid + mid - ll;
606 } else// MENU_TEXT_RIGHT)
607 sx = xmax + vo_font->charspace;
609 // We are after the start point -> go back
610 if(sx > xmin) {
611 for(n-- ; n <= 0 ; n--) {
612 unsigned char c = txt[n];
613 if(sx - vo_font->width[c] - vo_font->charspace < xmin) break;
614 sx -= vo_font->width[c]+vo_font->charspace;
616 } else { // We are before the start point -> go forward
617 for( ; sx < xmin && (&txt[n]) != line_end ; n++) {
618 unsigned char c = txt[n];
619 sx += vo_font->width[c]+vo_font->charspace;
622 txt = &txt[n]; // Jump to the new start char
623 } else {
624 if(align & MENU_TEXT_HCENTER)
625 sx = xmid - ll/2;
626 else
627 sx = xmax - 1 - ll;
629 } else {
630 for(sx = xrmin ; sx < xmin && txt != line_end ; txt++) {
631 unsigned char c = txt[n];
632 sx += vo_font->width[c]+vo_font->charspace;
636 while(sx < xmax && txt != line_end) {
637 int c=utf8_get_char((const char**)&txt);
638 font = vo_font->font[c];
639 if(font >= 0) {
640 int cs = (vo_font->pic_a[font]->h - vo_font->height) / 2;
641 if ((sx + vo_font->width[c] <= xmax) && (sy + vo_font->height <= ymax) )
642 draw_alpha(vo_font->width[c], vo_font->height,
643 vo_font->pic_b[font]->bmp+vo_font->start[c] +
644 cs * vo_font->pic_a[font]->w,
645 vo_font->pic_a[font]->bmp+vo_font->start[c] +
646 cs * vo_font->pic_a[font]->w,
647 vo_font->pic_a[font]->w,
648 mpi->planes[0] + sy * mpi->stride[0] + sx * (mpi->bpp>>3),
649 mpi->stride[0]);
650 // else
651 //printf("Can't draw '%c'\n",c);
653 sx+=vo_font->width[c]+vo_font->charspace;
655 txt = line_end;
656 if(txt[0] == '\0') break;
657 sy += vo_font->height + vspace;
661 int menu_text_length(char* txt) {
662 int l = 0;
663 render_txt(txt);
664 while (*txt) {
665 int c=utf8_get_char((const char**)&txt);
666 l += vo_font->width[c]+vo_font->charspace;
668 return l - vo_font->charspace;
671 void menu_text_size(char* txt,int max_width, int vspace, int warp, int* _w, int* _h) {
672 int l = 1, i = 0;
673 int w = 0;
675 render_txt(txt);
676 while (*txt) {
677 int c=utf8_get_char((const char**)&txt);
678 if(c == '\n' || (warp && i + vo_font->width[c] >= max_width)) {
679 i -= vo_font->charspace;
680 if (i > w) w = i;
681 if(*txt)
682 l++;
683 i = 0;
684 if(c == '\n') continue;
686 i += vo_font->width[c]+vo_font->charspace;
688 if (i > 0) {
689 i -= vo_font->charspace;
690 if (i > w) w = i;
693 *_w = w;
694 *_h = (l-1) * (vo_font->height + vspace) + vo_font->height;
698 int menu_text_num_lines(char* txt, int max_width) {
699 int l = 1, i = 0;
700 render_txt(txt);
701 while (*txt) {
702 int c=utf8_get_char((const char**)&txt);
703 if(c == '\n' || i + vo_font->width[c] > max_width) {
704 l++;
705 i = 0;
706 if(c == '\n') continue;
708 i += vo_font->width[c]+vo_font->charspace;
710 return l;
713 char* menu_text_get_next_line(char* txt, int max_width) {
714 int i = 0;
715 render_txt(txt);
716 while (*txt) {
717 int c=utf8_get_char((const char**)&txt);
718 if(c == '\n') {
719 txt++;
720 break;
722 i += vo_font->width[c];
723 if(i >= max_width)
724 break;
725 i += vo_font->charspace;
727 return txt;
731 void menu_draw_box(mp_image_t* mpi,unsigned char grey,unsigned char alpha, int x, int y, int w, int h) {
732 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
733 int g;
735 if(!draw_alpha) {
736 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
737 return;
740 if(x > mpi->w || y > mpi->h) return;
742 if(x < 0) w += x, x = 0;
743 if(x+w > mpi->w) w = mpi->w-x;
744 if(y < 0) h += y, y = 0;
745 if(y+h > mpi->h) h = mpi->h-y;
747 g = ((256-alpha)*grey)>>8;
748 if(g < 1) g = 1;
751 int stride = (w+7)&(~7); // round to 8
752 char pic[stride*h],pic_alpha[stride*h];
753 memset(pic,g,stride*h);
754 memset(pic_alpha,alpha,stride*h);
755 draw_alpha(w,h,pic,pic_alpha,stride,
756 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
757 mpi->stride[0]);