Raise LIBASS_VERSION, forgotten in r31293.
[mplayer/glamo.git] / libmenu / menu.c
bloba1fbe5787643b891801e54d3af4252ac80f5ee9a
1 /*
2 * This file is part of MPlayer.
4 * MPlayer is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
9 * MPlayer is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License along
15 * with MPlayer; if not, write to the Free Software Foundation, Inc.,
16 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 #include "config.h"
20 #include "mp_msg.h"
21 #include "help_mp.h"
23 #include <stdlib.h>
24 #include <stdio.h>
25 #include <string.h>
26 #include <fcntl.h>
27 #include <unistd.h>
29 #include "libvo/osd.h"
30 #include "libvo/font_load.h"
31 #include "libvo/sub.h"
32 #include "osdep/keycodes.h"
33 #include "asxparser.h"
34 #include "stream/stream.h"
35 #include "input/input.h"
37 #include "libmpcodecs/img_format.h"
38 #include "libmpcodecs/mp_image.h"
39 #include "m_option.h"
40 #include "m_struct.h"
41 #include "menu.h"
43 extern menu_info_t menu_info_cmdlist;
44 extern menu_info_t menu_info_chapsel;
45 extern menu_info_t menu_info_pt;
46 extern menu_info_t menu_info_filesel;
47 extern menu_info_t menu_info_txt;
48 extern menu_info_t menu_info_console;
49 extern menu_info_t menu_info_pref;
50 extern menu_info_t menu_info_dvbsel;
53 menu_info_t* menu_info_list[] = {
54 &menu_info_pt,
55 &menu_info_cmdlist,
56 &menu_info_chapsel,
57 &menu_info_filesel,
58 &menu_info_txt,
59 &menu_info_console,
60 #ifdef CONFIG_DVBIN
61 &menu_info_dvbsel,
62 #endif
63 &menu_info_pref,
64 NULL
67 typedef struct key_cmd_s {
68 int key;
69 char *cmd;
70 } key_cmd_t;
72 typedef struct menu_cmd_bindings_s {
73 char *name;
74 key_cmd_t *bindings;
75 int binding_num;
76 struct menu_cmd_bindings_s *parent;
77 } menu_cmd_bindings_t;
79 struct menu_def_st {
80 char* name;
81 menu_info_t* type;
82 void* cfg;
83 char* args;
86 double menu_mouse_x = -1.0;
87 double menu_mouse_y = -1.0;
88 int menu_mouse_pos_updated = 0;
90 static struct MPContext *menu_ctx = NULL;
91 static menu_def_t* menu_list = NULL;
92 static int menu_count = 0;
93 static menu_cmd_bindings_t *cmd_bindings = NULL;
94 static int cmd_bindings_num = 0;
97 static menu_cmd_bindings_t *get_cmd_bindings(const char *name)
99 int i;
100 for (i = 0; i < cmd_bindings_num; ++i)
101 if (!strcasecmp(cmd_bindings[i].name, name))
102 return &cmd_bindings[i];
103 return NULL;
106 static int menu_parse_config(char* buffer) {
107 char *element,*body, **attribs, *name;
108 menu_info_t* minfo = NULL;
109 int r,i;
110 ASX_Parser_t* parser = asx_parser_new();
112 while(1) {
113 r = asx_get_element(parser,&buffer,&element,&body,&attribs);
114 if(r < 0) {
115 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,parser->line);
116 asx_parser_free(parser);
117 return 0;
118 } else if(r == 0) {
119 asx_parser_free(parser);
120 return 1;
122 // Has it a name ?
123 name = asx_get_attrib("name",attribs);
124 if(!name) {
125 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuDefinitionsNeedANameAttrib,parser->line);
126 free(element);
127 if(body) free(body);
128 asx_free_attribs(attribs);
129 continue;
132 if (!strcasecmp(element, "keybindings")) {
133 menu_cmd_bindings_t *bindings = cmd_bindings;
134 char *parent_bindings;
135 cmd_bindings = realloc(cmd_bindings,
136 (cmd_bindings_num+1)*sizeof(menu_cmd_bindings_t));
137 for (i = 0; i < cmd_bindings_num; ++i)
138 if (cmd_bindings[i].parent)
139 cmd_bindings[i].parent = cmd_bindings[i].parent-bindings+cmd_bindings;
140 bindings = &cmd_bindings[cmd_bindings_num];
141 memset(bindings, 0, sizeof(menu_cmd_bindings_t));
142 bindings->name = name;
143 parent_bindings = asx_get_attrib("parent",attribs);
144 if (parent_bindings) {
145 bindings->parent = get_cmd_bindings(parent_bindings);
146 free(parent_bindings);
148 free(element);
149 asx_free_attribs(attribs);
150 if (body) {
151 char *bd = body;
152 char *b, *key, *cmd;
153 int keycode;
154 for(;;) {
155 r = asx_get_element(parser,&bd,&element,&b,&attribs);
156 if(r < 0) {
157 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_SyntaxErrorAtLine,
158 parser->line);
159 free(body);
160 asx_parser_free(parser);
161 return 0;
163 if(r == 0)
164 break;
165 if (!strcasecmp(element, "binding")) {
166 key = asx_get_attrib("key",attribs);
167 cmd = asx_get_attrib("cmd",attribs);
168 if (key && (keycode = mp_input_get_key_from_name(key)) >= 0) {
169 keycode &= ~MP_NO_REPEAT_KEY;
170 mp_msg(MSGT_GLOBAL,MSGL_V,
171 "[libmenu] got keybinding element %d %s=>[%s].\n",
172 keycode, key, cmd ? cmd : "");
173 bindings->bindings = realloc(bindings->bindings,
174 (bindings->binding_num+1)*sizeof(key_cmd_t));
175 bindings->bindings[bindings->binding_num].key = keycode;
176 bindings->bindings[bindings->binding_num].cmd = cmd;
177 ++bindings->binding_num;
179 else
180 free(cmd);
181 free(key);
183 free(element);
184 asx_free_attribs(attribs);
185 free(b);
187 free(body);
189 ++cmd_bindings_num;
190 continue;
192 // Try to find this menu type in our list
193 for(i = 0, minfo = NULL ; menu_info_list[i] ; i++) {
194 if(strcasecmp(element,menu_info_list[i]->name) == 0) {
195 minfo = menu_info_list[i];
196 break;
199 // Got it : add this to our list
200 if(minfo) {
201 menu_list = realloc(menu_list,(menu_count+2)*sizeof(menu_def_t));
202 menu_list[menu_count].name = name;
203 menu_list[menu_count].type = minfo;
204 menu_list[menu_count].cfg = m_struct_alloc(&minfo->priv_st);
205 menu_list[menu_count].args = body;
206 // Setup the attribs
207 for(i = 0 ; attribs[2*i] ; i++) {
208 if(strcasecmp(attribs[2*i],"name") == 0) continue;
209 if(!m_struct_set(&minfo->priv_st,menu_list[menu_count].cfg,attribs[2*i], attribs[2*i+1]))
210 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_BadAttrib,attribs[2*i],attribs[2*i+1],
211 name,parser->line);
213 menu_count++;
214 memset(&menu_list[menu_count],0,sizeof(menu_def_t));
215 } else {
216 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnknownMenuType,element,parser->line);
217 free(name);
218 if(body) free(body);
221 free(element);
222 asx_free_attribs(attribs);
228 /// This will build the menu_defs list from the cfg file
229 #define BUF_STEP 1024
230 #define BUF_MIN 128
231 #define BUF_MAX BUF_STEP*1024
232 int menu_init(struct MPContext *mpctx, char* cfg_file) {
233 char* buffer = NULL;
234 int bl = BUF_STEP, br = 0;
235 int f, fd;
236 #ifndef CONFIG_FREETYPE
237 if(vo_font == NULL)
238 return 0;
239 #endif
240 fd = open(cfg_file, O_RDONLY);
241 if(fd < 0) {
242 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_CantOpenConfigFile,cfg_file);
243 return 0;
245 buffer = malloc(bl);
246 while(1) {
247 int r;
248 if(bl - br < BUF_MIN) {
249 if(bl >= BUF_MAX) {
250 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsTooBig,BUF_MAX/1024);
251 close(fd);
252 free(buffer);
253 return 0;
255 bl += BUF_STEP;
256 buffer = realloc(buffer,bl);
258 r = read(fd,buffer+br,bl-br);
259 if(r == 0) break;
260 br += r;
262 if(!br) {
263 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_ConfigFileIsEmpty);
264 return 0;
266 buffer[br-1] = '\0';
268 close(fd);
270 menu_ctx = mpctx;
271 f = menu_parse_config(buffer);
272 free(buffer);
273 return f;
276 // Destroy all this stuff
277 void menu_uninit(void) {
278 int i;
279 for(i = 0 ; menu_list && menu_list[i].name ; i++) {
280 free(menu_list[i].name);
281 m_struct_free(&menu_list[i].type->priv_st,menu_list[i].cfg);
282 if(menu_list[i].args) free(menu_list[i].args);
284 free(menu_list);
285 menu_count = 0;
286 for (i = 0; i < cmd_bindings_num; ++i) {
287 free(cmd_bindings[i].name);
288 while(cmd_bindings[i].binding_num > 0)
289 free(cmd_bindings[i].bindings[--cmd_bindings[i].binding_num].cmd);
290 free(cmd_bindings[i].bindings);
292 free(cmd_bindings);
295 /// Default read_key function
296 int menu_dflt_read_key(menu_t* menu,int cmd) {
297 int i;
298 menu_cmd_bindings_t *bindings = get_cmd_bindings(menu->type->name);
299 if (!bindings)
300 bindings = get_cmd_bindings(menu->type->type->name);
301 if (!bindings)
302 bindings = get_cmd_bindings("default");
303 while (bindings) {
304 for (i = 0; i < bindings->binding_num; ++i) {
305 if (bindings->bindings[i].key == cmd) {
306 if (bindings->bindings[i].cmd)
307 mp_input_parse_and_queue_cmds(bindings->bindings[i].cmd);
308 return 1;
311 bindings = bindings->parent;
313 return 0;
316 menu_t* menu_open(char *name) {
317 menu_t* m;
318 int i;
320 for(i = 0 ; menu_list[i].name != NULL ; i++) {
321 if(strcmp(name,menu_list[i].name) == 0)
322 break;
324 if(menu_list[i].name == NULL) {
325 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuNotFound,name);
326 return NULL;
328 m = calloc(1,sizeof(menu_t));
329 m->priv_st = &(menu_list[i].type->priv_st);
330 m->priv = m_struct_copy(m->priv_st,menu_list[i].cfg);
331 m->ctx = menu_ctx;
332 m->type = &menu_list[i];
333 if(menu_list[i].type->open(m,menu_list[i].args))
334 return m;
335 if(m->priv)
336 m_struct_free(m->priv_st,m->priv);
337 free(m);
338 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_MenuInitFailed,name);
339 return NULL;
342 void menu_draw(menu_t* menu,mp_image_t* mpi) {
343 if(menu->show && menu->draw)
344 menu->draw(menu,mpi);
347 void menu_update_mouse_pos(double x, double y) {
348 menu_mouse_x = x;
349 menu_mouse_y = y;
350 menu_mouse_pos_updated = 1;
353 void menu_read_cmd(menu_t* menu,int cmd) {
354 if(menu->read_cmd)
355 menu->read_cmd(menu,cmd);
358 void menu_close(menu_t* menu) {
359 if(menu->close)
360 menu->close(menu);
361 if(menu->priv)
362 m_struct_free(menu->priv_st,menu->priv);
363 free(menu);
366 int menu_read_key(menu_t* menu,int cmd) {
367 if(menu->read_key)
368 return menu->read_key(menu,cmd);
369 else
370 return menu_dflt_read_key(menu,cmd);
373 ///////////////////////////// Helpers ////////////////////////////////////
375 typedef void (*draw_alpha_f)(int w,int h, unsigned char* src, unsigned char *srca, int srcstride, unsigned char* dstbase,int dststride);
377 inline static draw_alpha_f get_draw_alpha(uint32_t fmt) {
378 switch(fmt) {
379 case IMGFMT_BGR12:
380 case IMGFMT_RGB12:
381 return vo_draw_alpha_rgb12;
382 case IMGFMT_BGR15:
383 case IMGFMT_RGB15:
384 return vo_draw_alpha_rgb15;
385 case IMGFMT_BGR16:
386 case IMGFMT_RGB16:
387 return vo_draw_alpha_rgb16;
388 case IMGFMT_BGR24:
389 case IMGFMT_RGB24:
390 return vo_draw_alpha_rgb24;
391 case IMGFMT_BGR32:
392 case IMGFMT_RGB32:
393 return vo_draw_alpha_rgb32;
394 case IMGFMT_YV12:
395 case IMGFMT_I420:
396 case IMGFMT_IYUV:
397 case IMGFMT_YVU9:
398 case IMGFMT_IF09:
399 case IMGFMT_Y800:
400 case IMGFMT_Y8:
401 return vo_draw_alpha_yv12;
402 case IMGFMT_YUY2:
403 return vo_draw_alpha_yuy2;
404 case IMGFMT_UYVY:
405 return vo_draw_alpha_uyvy;
408 return NULL;
411 // return the real height of a char:
412 static inline int get_height(int c,int h){
413 int font;
414 if ((font=vo_font->font[c])>=0)
415 if(h<vo_font->pic_a[font]->h) h=vo_font->pic_a[font]->h;
416 return h;
419 static void render_txt(char *txt)
421 while (*txt) {
422 int c = utf8_get_char((const char**)&txt);
423 render_one_glyph(vo_font, c);
427 #ifdef CONFIG_FRIBIDI
428 #include <fribidi/fribidi.h>
429 #include "libavutil/common.h"
430 char *menu_fribidi_charset = NULL;
431 int menu_flip_hebrew = 0;
432 int menu_fribidi_flip_commas = 0;
434 static char *menu_fribidi(char *txt)
436 static int char_set_num = -1;
437 static FriBidiChar *logical, *visual;
438 static size_t buffer_size = 1024;
439 static char *outputstr;
441 FriBidiCharType base;
442 fribidi_boolean log2vis;
443 size_t len;
445 if (menu_flip_hebrew) {
446 len = strlen(txt);
447 if (char_set_num == -1) {
448 fribidi_set_mirroring (1);
449 fribidi_set_reorder_nsm (0);
450 char_set_num = fribidi_parse_charset("UTF-8");
451 buffer_size = FFMAX(1024,len+1);
452 logical = malloc(buffer_size);
453 visual = malloc(buffer_size);
454 outputstr = malloc(buffer_size);
455 } else if (len+1 > buffer_size) {
456 buffer_size = len+1;
457 logical = realloc(logical, buffer_size);
458 visual = realloc(visual, buffer_size);
459 outputstr = realloc(outputstr, buffer_size);
461 len = fribidi_charset_to_unicode (char_set_num, txt, len, logical);
462 base = menu_fribidi_flip_commas?FRIBIDI_TYPE_ON:FRIBIDI_TYPE_L;
463 log2vis = fribidi_log2vis (logical, len, &base, visual, NULL, NULL, NULL);
464 if (log2vis) {
465 len = fribidi_remove_bidi_marks (visual, len, NULL, NULL, NULL);
466 fribidi_unicode_to_charset (char_set_num, visual, len, outputstr);
467 return outputstr;
470 return txt;
472 #endif
474 void menu_draw_text(mp_image_t* mpi,char* txt, int x, int y) {
475 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
476 int font;
478 if(!draw_alpha) {
479 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
480 return;
483 #ifdef CONFIG_FRIBIDI
484 txt = menu_fribidi(txt);
485 #endif
486 render_txt(txt);
488 while (*txt) {
489 int c=utf8_get_char((const char**)&txt);
490 if ((font=vo_font->font[c])>=0 && (x + vo_font->width[c] <= mpi->w) && (y + vo_font->pic_a[font]->h <= mpi->h))
491 draw_alpha(vo_font->width[c], vo_font->pic_a[font]->h,
492 vo_font->pic_b[font]->bmp+vo_font->start[c],
493 vo_font->pic_a[font]->bmp+vo_font->start[c],
494 vo_font->pic_a[font]->w,
495 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
496 mpi->stride[0]);
497 x+=vo_font->width[c]+vo_font->charspace;
502 void menu_draw_text_full(mp_image_t* mpi,char* txt,
503 int x, int y,int w, int h,
504 int vspace, int warp, int align, int anchor) {
505 int need_w,need_h;
506 int sy, ymin, ymax;
507 int sx, xmin, xmax, xmid, xrmin;
508 int ll = 0;
509 int font;
510 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
512 if(!draw_alpha) {
513 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
514 return;
517 #ifdef CONFIG_FRIBIDI
518 txt = menu_fribidi(txt);
519 #endif
520 render_txt(txt);
522 if(x > mpi->w || y > mpi->h)
523 return;
525 if(anchor & MENU_TEXT_VCENTER) {
526 if(h <= 0) h = mpi->h;
527 ymin = y - h/2;
528 ymax = y + h/2;
529 } else if(anchor & MENU_TEXT_BOT) {
530 if(h <= 0) h = mpi->h - y;
531 ymin = y - h;
532 ymax = y;
533 } else {
534 if(h <= 0) h = mpi->h - y;
535 ymin = y;
536 ymax = y + h;
539 if(anchor & MENU_TEXT_HCENTER) {
540 if(w <= 0) w = mpi->w;
541 xmin = x - w/2;
542 xmax = x + w/2;
543 } else if(anchor & MENU_TEXT_RIGHT) {
544 if(w <= 0) w = mpi->w -x;
545 xmin = x - w;
546 xmax = x;
547 } else {
548 if(w <= 0) w = mpi->w -x;
549 xmin = x;
550 xmax = x + w;
553 // How many space do we need to draw this ?
554 menu_text_size(txt,w,vspace,warp,&need_w,&need_h);
556 // Find the first line
557 if(align & MENU_TEXT_VCENTER)
558 sy = ymin + ((h - need_h)/2);
559 else if(align & MENU_TEXT_BOT)
560 sy = ymax - need_h - 1;
561 else
562 sy = y;
564 #if 0
565 // Find the first col
566 if(align & MENU_TEXT_HCENTER)
567 sx = xmin + ((w - need_w)/2);
568 else if(align & MENU_TEXT_RIGHT)
569 sx = xmax - need_w;
570 #endif
572 xmid = xmin + (xmax - xmin) / 2;
573 xrmin = xmin;
574 // Clamp the bb to the mpi size
575 if(ymin < 0) ymin = 0;
576 if(xmin < 0) xmin = 0;
577 if(ymax > mpi->h) ymax = mpi->h;
578 if(xmax > mpi->w) xmax = mpi->w;
580 // Jump some the beginnig text if needed
581 while(sy < ymin && *txt) {
582 int c=utf8_get_char((const char**)&txt);
583 if(c == '\n' || (warp && ll + vo_font->width[c] > w)) {
584 ll = 0;
585 sy += vo_font->height + vspace;
586 if(c == '\n') continue;
588 ll += vo_font->width[c]+vo_font->charspace;
590 if(*txt == '\0') // Nothing left to draw
591 return;
593 while(sy < ymax && *txt) {
594 char* line_end = NULL;
595 int n;
597 if(txt[0] == '\n') { // New line
598 sy += vo_font->height + vspace;
599 txt++;
600 continue;
603 // Get the length and end of this line
604 for(n = 0, ll = 0 ; txt[n] != '\0' && txt[n] != '\n' ; n++) {
605 unsigned char c = txt[n];
606 if(warp && ll + vo_font->width[c] > w) break;
607 ll += vo_font->width[c]+vo_font->charspace;
609 line_end = &txt[n];
610 ll -= vo_font->charspace;
613 if(align & (MENU_TEXT_HCENTER|MENU_TEXT_RIGHT)) {
614 // Too long line
615 if(ll > xmax-xmin) {
616 if(align & MENU_TEXT_HCENTER) {
617 int mid = ll/2;
618 // Find the middle point
619 for(n--, ll = 0 ; n <= 0 ; n--) {
620 ll += vo_font->width[(int)txt[n]]+vo_font->charspace;
621 if(ll - vo_font->charspace > mid) break;
623 ll -= vo_font->charspace;
624 sx = xmid + mid - ll;
625 } else// MENU_TEXT_RIGHT)
626 sx = xmax + vo_font->charspace;
628 // We are after the start point -> go back
629 if(sx > xmin) {
630 for(n-- ; n <= 0 ; n--) {
631 unsigned char c = txt[n];
632 if(sx - vo_font->width[c] - vo_font->charspace < xmin) break;
633 sx -= vo_font->width[c]+vo_font->charspace;
635 } else { // We are before the start point -> go forward
636 for( ; sx < xmin && (&txt[n]) != line_end ; n++) {
637 unsigned char c = txt[n];
638 sx += vo_font->width[c]+vo_font->charspace;
641 txt = &txt[n]; // Jump to the new start char
642 } else {
643 if(align & MENU_TEXT_HCENTER)
644 sx = xmid - ll/2;
645 else
646 sx = xmax - 1 - ll;
648 } else {
649 for(sx = xrmin ; sx < xmin && txt != line_end ; txt++) {
650 unsigned char c = txt[n];
651 sx += vo_font->width[c]+vo_font->charspace;
655 while(sx < xmax && txt != line_end) {
656 int c=utf8_get_char((const char**)&txt);
657 font = vo_font->font[c];
658 if(font >= 0) {
659 int cs = (vo_font->pic_a[font]->h - vo_font->height) / 2;
660 if ((sx + vo_font->width[c] <= xmax) && (sy + vo_font->height <= ymax) )
661 draw_alpha(vo_font->width[c], vo_font->height,
662 vo_font->pic_b[font]->bmp+vo_font->start[c] +
663 cs * vo_font->pic_a[font]->w,
664 vo_font->pic_a[font]->bmp+vo_font->start[c] +
665 cs * vo_font->pic_a[font]->w,
666 vo_font->pic_a[font]->w,
667 mpi->planes[0] + sy * mpi->stride[0] + sx * (mpi->bpp>>3),
668 mpi->stride[0]);
669 // else
670 //printf("Can't draw '%c'\n",c);
672 sx+=vo_font->width[c]+vo_font->charspace;
674 txt = line_end;
675 if(txt[0] == '\0') break;
676 sy += vo_font->height + vspace;
680 int menu_text_length(char* txt) {
681 int l = 0;
682 render_txt(txt);
683 while (*txt) {
684 int c=utf8_get_char((const char**)&txt);
685 l += vo_font->width[c]+vo_font->charspace;
687 return l - vo_font->charspace;
690 void menu_text_size(char* txt,int max_width, int vspace, int warp, int* _w, int* _h) {
691 int l = 1, i = 0;
692 int w = 0;
694 render_txt(txt);
695 while (*txt) {
696 int c=utf8_get_char((const char**)&txt);
697 if(c == '\n' || (warp && i + vo_font->width[c] >= max_width)) {
698 i -= vo_font->charspace;
699 if (i > w) w = i;
700 if(*txt)
701 l++;
702 i = 0;
703 if(c == '\n') continue;
705 i += vo_font->width[c]+vo_font->charspace;
707 if (i > 0) {
708 i -= vo_font->charspace;
709 if (i > w) w = i;
712 *_w = w;
713 *_h = (l-1) * (vo_font->height + vspace) + vo_font->height;
717 int menu_text_num_lines(char* txt, int max_width) {
718 int l = 1, i = 0;
719 render_txt(txt);
720 while (*txt) {
721 int c=utf8_get_char((const char**)&txt);
722 if(c == '\n' || i + vo_font->width[c] > max_width) {
723 l++;
724 i = 0;
725 if(c == '\n') continue;
727 i += vo_font->width[c]+vo_font->charspace;
729 return l;
732 static char* menu_text_get_next_line(char* txt, int max_width)
734 int i = 0;
735 render_txt(txt);
736 while (*txt) {
737 int c=utf8_get_char((const char**)&txt);
738 if(c == '\n') {
739 txt++;
740 break;
742 i += vo_font->width[c];
743 if(i >= max_width)
744 break;
745 i += vo_font->charspace;
747 return txt;
751 void menu_draw_box(mp_image_t* mpi,unsigned char grey,unsigned char alpha, int x, int y, int w, int h) {
752 draw_alpha_f draw_alpha = get_draw_alpha(mpi->imgfmt);
753 int g;
755 if(!draw_alpha) {
756 mp_msg(MSGT_GLOBAL,MSGL_WARN,MSGTR_LIBMENU_UnsupportedOutformat);
757 return;
760 if(x > mpi->w || y > mpi->h) return;
762 if(x < 0) w += x, x = 0;
763 if(x+w > mpi->w) w = mpi->w-x;
764 if(y < 0) h += y, y = 0;
765 if(y+h > mpi->h) h = mpi->h-y;
767 g = ((256-alpha)*grey)>>8;
768 if(g < 1) g = 1;
771 int stride = (w+7)&(~7); // round to 8
772 char pic[stride*h],pic_alpha[stride*h];
773 memset(pic,g,stride*h);
774 memset(pic_alpha,alpha,stride*h);
775 draw_alpha(w,h,pic,pic_alpha,stride,
776 mpi->planes[0] + y * mpi->stride[0] + x * (mpi->bpp>>3),
777 mpi->stride[0]);