Simplify handling of "flags" parameter
[mplayer/glamo.git] / libmenu / menu.c
blob41f33801fc9236eaf12127aae63ea3f9661d2ca7
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 extern menu_info_t menu_info_dvbsel;
36 menu_info_t* menu_info_list[] = {
37 &menu_info_pt,
38 &menu_info_cmdlist,
39 &menu_info_chapsel,
40 &menu_info_filesel,
41 &menu_info_txt,
42 &menu_info_console,
43 #ifdef CONFIG_DVBIN
44 &menu_info_dvbsel,
45 #endif
46 &menu_info_pref,
47 NULL
50 typedef struct key_cmd_s {
51 int key;
52 char *cmd;
53 } key_cmd_t;
55 typedef struct menu_cmd_bindings_s {
56 char *name;
57 key_cmd_t *bindings;
58 int binding_num;
59 struct menu_cmd_bindings_s *parent;
60 } menu_cmd_bindings_t;
62 struct menu_def_st {
63 char* name;
64 menu_info_t* type;
65 void* cfg;
66 char* args;
69 double menu_mouse_x = -1.0;
70 double menu_mouse_y = -1.0;
71 int menu_mouse_pos_updated = 0;
73 static struct MPContext *menu_ctx = NULL;
74 static menu_def_t* menu_list = NULL;
75 static int menu_count = 0;
76 static menu_cmd_bindings_t *cmd_bindings = NULL;
77 static int cmd_bindings_num = 0;
80 menu_cmd_bindings_t *get_cmd_bindings(const char *name) {
81 int i;
82 for (i = 0; i < cmd_bindings_num; ++i)
83 if (!strcasecmp(cmd_bindings[i].name, name))
84 return &cmd_bindings[i];
85 return NULL;
88 static int menu_parse_config(char* buffer) {
89 char *element,*body, **attribs, *name;
90 menu_info_t* minfo = NULL;
91 int r,i;
92 ASX_Parser_t* parser = asx_parser_new();
94 while(1) {
95 r = asx_get_element(parser,&buffer,&element,&body,&attribs);
96 if(r < 0) {
97 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,parser->line);
98 asx_parser_free(parser);
99 return 0;
100 } else if(r == 0) {
101 asx_parser_free(parser);
102 return 1;
104 // Has it a name ?
105 name = asx_get_attrib("name",attribs);
106 if(!name) {
107 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuDefinitionsNeedANameAttrib,parser->line);
108 free(element);
109 if(body) free(body);
110 asx_free_attribs(attribs);
111 continue;
114 if (!strcasecmp(element, "keybindings")) {
115 menu_cmd_bindings_t *bindings = cmd_bindings;
116 char *parent_bindings;
117 cmd_bindings = realloc(cmd_bindings,
118 (cmd_bindings_num+1)*sizeof(menu_cmd_bindings_t));
119 for (i = 0; i < cmd_bindings_num; ++i)
120 if (cmd_bindings[i].parent)
121 cmd_bindings[i].parent = cmd_bindings[i].parent-bindings+cmd_bindings;
122 bindings = &cmd_bindings[cmd_bindings_num];
123 memset(bindings, 0, sizeof(menu_cmd_bindings_t));
124 bindings->name = name;
125 parent_bindings = asx_get_attrib("parent",attribs);
126 if (parent_bindings) {
127 bindings->parent = get_cmd_bindings(parent_bindings);
128 free(parent_bindings);
130 free(element);
131 asx_free_attribs(attribs);
132 if (body) {
133 char *bd = body;
134 char *b, *key, *cmd;
135 int keycode;
136 for(;;) {
137 r = asx_get_element(parser,&bd,&element,&b,&attribs);
138 if(r < 0) {
139 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,
140 parser->line);
141 free(body);
142 asx_parser_free(parser);
143 return 0;
145 if(r == 0)
146 break;
147 if (!strcasecmp(element, "binding")) {
148 key = asx_get_attrib("key",attribs);
149 cmd = asx_get_attrib("cmd",attribs);
150 if (key && (keycode = mp_input_get_key_from_name(key)) >= 0) {
151 keycode &= ~MP_NO_REPEAT_KEY;
152 mp_msg(MSGT_GLOBAL,MSGL_V,
153 "[libmenu] got keybinding element %d %s=>[%s].\n",
154 keycode, key, cmd ? cmd : "");
155 bindings->bindings = realloc(bindings->bindings,
156 (bindings->binding_num+1)*sizeof(key_cmd_t));
157 bindings->bindings[bindings->binding_num].key = keycode;
158 bindings->bindings[bindings->binding_num].cmd = cmd;
159 ++bindings->binding_num;
161 else
162 free(cmd);
163 free(key);
165 free(element);
166 asx_free_attribs(attribs);
167 free(b);
169 free(body);
171 ++cmd_bindings_num;
172 continue;
174 // Try to find this menu type in our list
175 for(i = 0, minfo = NULL ; menu_info_list[i] ; i++) {
176 if(strcasecmp(element,menu_info_list[i]->name) == 0) {
177 minfo = menu_info_list[i];
178 break;
181 // Got it : add this to our list
182 if(minfo) {
183 menu_list = realloc(menu_list,(menu_count+2)*sizeof(menu_def_t));
184 menu_list[menu_count].name = name;
185 menu_list[menu_count].type = minfo;
186 menu_list[menu_count].cfg = m_struct_alloc(&minfo->priv_st);
187 menu_list[menu_count].args = body;
188 // Setup the attribs
189 for(i = 0 ; attribs[2*i] ; i++) {
190 if(strcasecmp(attribs[2*i],"name") == 0) continue;
191 if(!m_struct_set(&minfo->priv_st,menu_list[menu_count].cfg,attribs[2*i], attribs[2*i+1]))
192 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_BadAttrib,attribs[2*i],attribs[2*i+1],
193 name,parser->line);
195 menu_count++;
196 memset(&menu_list[menu_count],0,sizeof(menu_def_t));
197 } else {
198 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnknownMenuType,element,parser->line);
199 free(name);
200 if(body) free(body);
203 free(element);
204 asx_free_attribs(attribs);
210 /// This will build the menu_defs list from the cfg file
211 #define BUF_STEP 1024
212 #define BUF_MIN 128
213 #define BUF_MAX BUF_STEP*1024
214 int menu_init(struct MPContext *mpctx, char* cfg_file) {
215 char* buffer = NULL;
216 int bl = BUF_STEP, br = 0;
217 int f, fd;
218 #ifndef CONFIG_FREETYPE
219 if(vo_font == NULL)
220 return 0;
221 #endif
222 fd = open(cfg_file, O_RDONLY);
223 if(fd < 0) {
224 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_CantOpenConfigFile,cfg_file);
225 return 0;
227 buffer = malloc(bl);
228 while(1) {
229 int r;
230 if(bl - br < BUF_MIN) {
231 if(bl >= BUF_MAX) {
232 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsTooBig,BUF_MAX/1024);
233 close(fd);
234 free(buffer);
235 return 0;
237 bl += BUF_STEP;
238 buffer = realloc(buffer,bl);
240 r = read(fd,buffer+br,bl-br);
241 if(r == 0) break;
242 br += r;
244 if(!br) {
245 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsEmpty);
246 return 0;
248 buffer[br-1] = '\0';
250 close(fd);
252 menu_ctx = mpctx;
253 f = menu_parse_config(buffer);
254 free(buffer);
255 return f;
258 // Destroy all this stuff
259 void menu_uninit(void) {
260 int i;
261 for(i = 0 ; menu_list && menu_list[i].name ; i++) {
262 free(menu_list[i].name);
263 m_struct_free(&menu_list[i].type->priv_st,menu_list[i].cfg);
264 if(menu_list[i].args) free(menu_list[i].args);
266 free(menu_list);
267 menu_count = 0;
268 for (i = 0; i < cmd_bindings_num; ++i) {
269 free(cmd_bindings[i].name);
270 while(cmd_bindings[i].binding_num > 0)
271 free(cmd_bindings[i].bindings[--cmd_bindings[i].binding_num].cmd);
272 free(cmd_bindings[i].bindings);
274 free(cmd_bindings);
277 /// Default read_key function
278 int menu_dflt_read_key(menu_t* menu,int cmd) {
279 int i;
280 menu_cmd_bindings_t *bindings = get_cmd_bindings(menu->type->name);
281 if (!bindings)
282 bindings = get_cmd_bindings(menu->type->type->name);
283 if (!bindings)
284 bindings = get_cmd_bindings("default");
285 while (bindings) {
286 for (i = 0; i < bindings->binding_num; ++i) {
287 if (bindings->bindings[i].key == cmd) {
288 if (bindings->bindings[i].cmd)
289 mp_input_parse_and_queue_cmds(bindings->bindings[i].cmd);
290 return 1;
293 bindings = bindings->parent;
295 return 0;
298 menu_t* menu_open(char *name) {
299 menu_t* m;
300 int i;
302 for(i = 0 ; menu_list[i].name != NULL ; i++) {
303 if(strcmp(name,menu_list[i].name) == 0)
304 break;
306 if(menu_list[i].name == NULL) {
307 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuNotFound,name);
308 return NULL;
310 m = calloc(1,sizeof(menu_t));
311 m->priv_st = &(menu_list[i].type->priv_st);
312 m->priv = m_struct_copy(m->priv_st,menu_list[i].cfg);
313 m->ctx = menu_ctx;
314 m->type = &menu_list[i];
315 if(menu_list[i].type->open(m,menu_list[i].args))
316 return m;
317 if(m->priv)
318 m_struct_free(m->priv_st,m->priv);
319 free(m);
320 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuInitFailed,name);
321 return NULL;
324 void menu_draw(menu_t* menu,mp_image_t* mpi) {
325 if(menu->show && menu->draw)
326 menu->draw(menu,mpi);
329 void menu_update_mouse_pos(double x, double y) {
330 menu_mouse_x = x;
331 menu_mouse_y = y;
332 menu_mouse_pos_updated = 1;
335 void menu_read_cmd(menu_t* menu,int cmd) {
336 if(menu->read_cmd)
337 menu->read_cmd(menu,cmd);
340 void menu_close(menu_t* menu) {
341 if(menu->close)
342 menu->close(menu);
343 if(menu->priv)
344 m_struct_free(menu->priv_st,menu->priv);
345 free(menu);
348 int menu_read_key(menu_t* menu,int cmd) {
349 if(menu->read_key)
350 return menu->read_key(menu,cmd);
351 else
352 return menu_dflt_read_key(menu,cmd);
355 ///////////////////////////// Helpers ////////////////////////////////////
357 typedef void (*draw_alpha_f)(int w,int h, unsigned char* src, unsigned char *srca, int srcstride, unsigned char* dstbase,int dststride);
359 inline static draw_alpha_f get_draw_alpha(uint32_t fmt) {
360 switch(fmt) {
361 case IMGFMT_BGR15:
362 case IMGFMT_RGB15:
363 return vo_draw_alpha_rgb15;
364 case IMGFMT_BGR16:
365 case IMGFMT_RGB16:
366 return vo_draw_alpha_rgb16;
367 case IMGFMT_BGR24:
368 case IMGFMT_RGB24:
369 return vo_draw_alpha_rgb24;
370 case IMGFMT_BGR32:
371 case IMGFMT_RGB32:
372 return vo_draw_alpha_rgb32;
373 case IMGFMT_YV12:
374 case IMGFMT_I420:
375 case IMGFMT_IYUV:
376 case IMGFMT_YVU9:
377 case IMGFMT_IF09:
378 case IMGFMT_Y800:
379 case IMGFMT_Y8:
380 return vo_draw_alpha_yv12;
381 case IMGFMT_YUY2:
382 return vo_draw_alpha_yuy2;
383 case IMGFMT_UYVY:
384 return vo_draw_alpha_uyvy;
387 return NULL;
390 // return the real height of a char:
391 static inline int get_height(int c,int h){
392 int font;
393 if ((font=vo_font->font[c])>=0)
394 if(h<vo_font->pic_a[font]->h) h=vo_font->pic_a[font]->h;
395 return h;
398 static void render_txt(char *txt)
400 while (*txt) {
401 int c = utf8_get_char((const char**)&txt);
402 render_one_glyph(vo_font, c);
406 #ifdef CONFIG_FRIBIDI
407 #include <fribidi/fribidi.h>
408 #include "libavutil/common.h"
409 char *menu_fribidi_charset = NULL;
410 int menu_flip_hebrew = 0;
411 int menu_fribidi_flip_commas = 0;
413 static char *menu_fribidi(char *txt)
415 static int char_set_num = -1;
416 static FriBidiChar *logical, *visual;
417 static size_t buffer_size = 1024;
418 static char *outputstr;
420 FriBidiCharType base;
421 fribidi_boolean log2vis;
422 size_t len;
424 if (menu_flip_hebrew) {
425 len = strlen(txt);
426 if (char_set_num == -1) {
427 fribidi_set_mirroring (1);
428 fribidi_set_reorder_nsm (0);
429 char_set_num = fribidi_parse_charset("UTF-8");
430 buffer_size = FFMAX(1024,len+1);
431 logical = malloc(buffer_size);
432 visual = malloc(buffer_size);
433 outputstr = malloc(buffer_size);
434 } else if (len+1 > buffer_size) {
435 buffer_size = len+1;
436 logical = realloc(logical, buffer_size);
437 visual = realloc(visual, buffer_size);
438 outputstr = realloc(outputstr, buffer_size);
440 len = fribidi_charset_to_unicode (char_set_num, txt, len, logical);
441 base = menu_fribidi_flip_commas?FRIBIDI_TYPE_ON:FRIBIDI_TYPE_L;
442 log2vis = fribidi_log2vis (logical, len, &base, visual, NULL, NULL, NULL);
443 if (log2vis) {
444 len = fribidi_remove_bidi_marks (visual, len, NULL, NULL, NULL);
445 fribidi_unicode_to_charset (char_set_num, visual, len, outputstr);
446 return outputstr;
449 return txt;
451 #endif
453 void menu_draw_text(mp_image_t* mpi,char* txt, int x, int y) {
454 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
455 int font;
457 if(!draw_alpha) {
458 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
459 return;
462 #ifdef CONFIG_FRIBIDI
463 txt = menu_fribidi(txt);
464 #endif
465 render_txt(txt);
467 while (*txt) {
468 int c=utf8_get_char((const char**)&txt);
469 if ((font=vo_font->font[c])>=0 && (x + vo_font->width[c] <= mpi->w) && (y + vo_font->pic_a[font]->h <= mpi->h))
470 draw_alpha(vo_font->width[c], vo_font->pic_a[font]->h,
471 vo_font->pic_b[font]->bmp+vo_font->start[c],
472 vo_font->pic_a[font]->bmp+vo_font->start[c],
473 vo_font->pic_a[font]->w,
474 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
475 mpi->stride[0]);
476 x+=vo_font->width[c]+vo_font->charspace;
481 void menu_draw_text_full(mp_image_t* mpi,char* txt,
482 int x, int y,int w, int h,
483 int vspace, int warp, int align, int anchor) {
484 int need_w,need_h;
485 int sy, ymin, ymax;
486 int sx, xmin, xmax, xmid, xrmin;
487 int ll = 0;
488 int font;
489 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
491 if(!draw_alpha) {
492 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
493 return;
496 #ifdef CONFIG_FRIBIDI
497 txt = menu_fribidi(txt);
498 #endif
499 render_txt(txt);
501 if(x > mpi->w || y > mpi->h)
502 return;
504 if(anchor & MENU_TEXT_VCENTER) {
505 if(h <= 0) h = mpi->h;
506 ymin = y - h/2;
507 ymax = y + h/2;
508 } else if(anchor & MENU_TEXT_BOT) {
509 if(h <= 0) h = mpi->h - y;
510 ymin = y - h;
511 ymax = y;
512 } else {
513 if(h <= 0) h = mpi->h - y;
514 ymin = y;
515 ymax = y + h;
518 if(anchor & MENU_TEXT_HCENTER) {
519 if(w <= 0) w = mpi->w;
520 xmin = x - w/2;
521 xmax = x + w/2;
522 } else if(anchor & MENU_TEXT_RIGHT) {
523 if(w <= 0) w = mpi->w -x;
524 xmin = x - w;
525 xmax = x;
526 } else {
527 if(w <= 0) w = mpi->w -x;
528 xmin = x;
529 xmax = x + w;
532 // How many space do we need to draw this ?
533 menu_text_size(txt,w,vspace,warp,&need_w,&need_h);
535 // Find the first line
536 if(align & MENU_TEXT_VCENTER)
537 sy = ymin + ((h - need_h)/2);
538 else if(align & MENU_TEXT_BOT)
539 sy = ymax - need_h - 1;
540 else
541 sy = y;
543 #if 0
544 // Find the first col
545 if(align & MENU_TEXT_HCENTER)
546 sx = xmin + ((w - need_w)/2);
547 else if(align & MENU_TEXT_RIGHT)
548 sx = xmax - need_w;
549 #endif
551 xmid = xmin + (xmax - xmin) / 2;
552 xrmin = xmin;
553 // Clamp the bb to the mpi size
554 if(ymin < 0) ymin = 0;
555 if(xmin < 0) xmin = 0;
556 if(ymax > mpi->h) ymax = mpi->h;
557 if(xmax > mpi->w) xmax = mpi->w;
559 // Jump some the beginnig text if needed
560 while(sy < ymin && *txt) {
561 int c=utf8_get_char((const char**)&txt);
562 if(c == '\n' || (warp && ll + vo_font->width[c] > w)) {
563 ll = 0;
564 sy += vo_font->height + vspace;
565 if(c == '\n') continue;
567 ll += vo_font->width[c]+vo_font->charspace;
569 if(*txt == '\0') // Nothing left to draw
570 return;
572 while(sy < ymax && *txt) {
573 char* line_end = NULL;
574 int n;
576 if(txt[0] == '\n') { // New line
577 sy += vo_font->height + vspace;
578 txt++;
579 continue;
582 // Get the length and end of this line
583 for(n = 0, ll = 0 ; txt[n] != '\0' && txt[n] != '\n' ; n++) {
584 unsigned char c = txt[n];
585 if(warp && ll + vo_font->width[c] > w) break;
586 ll += vo_font->width[c]+vo_font->charspace;
588 line_end = &txt[n];
589 ll -= vo_font->charspace;
592 if(align & (MENU_TEXT_HCENTER|MENU_TEXT_RIGHT)) {
593 // Too long line
594 if(ll > xmax-xmin) {
595 if(align & MENU_TEXT_HCENTER) {
596 int mid = ll/2;
597 // Find the middle point
598 for(n--, ll = 0 ; n <= 0 ; n--) {
599 ll += vo_font->width[(int)txt[n]]+vo_font->charspace;
600 if(ll - vo_font->charspace > mid) break;
602 ll -= vo_font->charspace;
603 sx = xmid + mid - ll;
604 } else// MENU_TEXT_RIGHT)
605 sx = xmax + vo_font->charspace;
607 // We are after the start point -> go back
608 if(sx > xmin) {
609 for(n-- ; n <= 0 ; n--) {
610 unsigned char c = txt[n];
611 if(sx - vo_font->width[c] - vo_font->charspace < xmin) break;
612 sx -= vo_font->width[c]+vo_font->charspace;
614 } else { // We are before the start point -> go forward
615 for( ; sx < xmin && (&txt[n]) != line_end ; n++) {
616 unsigned char c = txt[n];
617 sx += vo_font->width[c]+vo_font->charspace;
620 txt = &txt[n]; // Jump to the new start char
621 } else {
622 if(align & MENU_TEXT_HCENTER)
623 sx = xmid - ll/2;
624 else
625 sx = xmax - 1 - ll;
627 } else {
628 for(sx = xrmin ; sx < xmin && txt != line_end ; txt++) {
629 unsigned char c = txt[n];
630 sx += vo_font->width[c]+vo_font->charspace;
634 while(sx < xmax && txt != line_end) {
635 int c=utf8_get_char((const char**)&txt);
636 font = vo_font->font[c];
637 if(font >= 0) {
638 int cs = (vo_font->pic_a[font]->h - vo_font->height) / 2;
639 if ((sx + vo_font->width[c] <= xmax) && (sy + vo_font->height <= ymax) )
640 draw_alpha(vo_font->width[c], vo_font->height,
641 vo_font->pic_b[font]->bmp+vo_font->start[c] +
642 cs * vo_font->pic_a[font]->w,
643 vo_font->pic_a[font]->bmp+vo_font->start[c] +
644 cs * vo_font->pic_a[font]->w,
645 vo_font->pic_a[font]->w,
646 mpi->planes[0] + sy * mpi->stride[0] + sx * (mpi->bpp>>3),
647 mpi->stride[0]);
648 // else
649 //printf("Can't draw '%c'\n",c);
651 sx+=vo_font->width[c]+vo_font->charspace;
653 txt = line_end;
654 if(txt[0] == '\0') break;
655 sy += vo_font->height + vspace;
659 int menu_text_length(char* txt) {
660 int l = 0;
661 render_txt(txt);
662 while (*txt) {
663 int c=utf8_get_char((const char**)&txt);
664 l += vo_font->width[c]+vo_font->charspace;
666 return l - vo_font->charspace;
669 void menu_text_size(char* txt,int max_width, int vspace, int warp, int* _w, int* _h) {
670 int l = 1, i = 0;
671 int w = 0;
673 render_txt(txt);
674 while (*txt) {
675 int c=utf8_get_char((const char**)&txt);
676 if(c == '\n' || (warp && i + vo_font->width[c] >= max_width)) {
677 i -= vo_font->charspace;
678 if (i > w) w = i;
679 if(*txt)
680 l++;
681 i = 0;
682 if(c == '\n') continue;
684 i += vo_font->width[c]+vo_font->charspace;
686 if (i > 0) {
687 i -= vo_font->charspace;
688 if (i > w) w = i;
691 *_w = w;
692 *_h = (l-1) * (vo_font->height + vspace) + vo_font->height;
696 int menu_text_num_lines(char* txt, int max_width) {
697 int l = 1, i = 0;
698 render_txt(txt);
699 while (*txt) {
700 int c=utf8_get_char((const char**)&txt);
701 if(c == '\n' || i + vo_font->width[c] > max_width) {
702 l++;
703 i = 0;
704 if(c == '\n') continue;
706 i += vo_font->width[c]+vo_font->charspace;
708 return l;
711 char* menu_text_get_next_line(char* txt, int max_width) {
712 int i = 0;
713 render_txt(txt);
714 while (*txt) {
715 int c=utf8_get_char((const char**)&txt);
716 if(c == '\n') {
717 txt++;
718 break;
720 i += vo_font->width[c];
721 if(i >= max_width)
722 break;
723 i += vo_font->charspace;
725 return txt;
729 void menu_draw_box(mp_image_t* mpi,unsigned char grey,unsigned char alpha, int x, int y, int w, int h) {
730 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
731 int g;
733 if(!draw_alpha) {
734 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
735 return;
738 if(x > mpi->w || y > mpi->h) return;
740 if(x < 0) w += x, x = 0;
741 if(x+w > mpi->w) w = mpi->w-x;
742 if(y < 0) h += y, y = 0;
743 if(y+h > mpi->h) h = mpi->h-y;
745 g = ((256-alpha)*grey)>>8;
746 if(g < 1) g = 1;
749 int stride = (w+7)&(~7); // round to 8
750 char pic[stride*h],pic_alpha[stride*h];
751 memset(pic,g,stride*h);
752 memset(pic_alpha,alpha,stride*h);
753 draw_alpha(w,h,pic,pic_alpha,stride,
754 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
755 mpi->stride[0]);