13cc6816c3824bef2f8dc643133298db4ff08cd2
[goodguy/cinelerra.git] / cinelerra-5.1 / guicast / bcpopupmenu.C
1
2 /*
3  * CINELERRA
4  * Copyright (C) 2008 Adam Williams <broadcast at earthling dot net>
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  */
21
22 #include "bcmenubar.h"
23 #include "bcmenuitem.h"
24 #include "bcmenupopup.h"
25 #include "bcpixmap.h"
26 #include "bcpopupmenu.h"
27 #include "bcresources.h"
28 #include "bcsignals.h"
29 #include "bccolors.h"
30 #include "fonts.h"
31 #include <string.h>
32 #include "vframe.h"
33
34 #define BUTTON_UP 0
35 #define BUTTON_HI 1
36 #define BUTTON_DN 2
37 #define TOTAL_IMAGES 3
38
39
40 #define TRIANGLE_W 10
41 #define TRIANGLE_H 10
42
43
44 BC_PopupMenu::BC_PopupMenu(int x, int y, int w, const char *text,
45                 int use_title, VFrame **data, int margin)
46  : BC_SubWindow(x, y, 0, 0, -1)
47 {
48         highlighted = popup_down = 0;
49         menu_popup = 0;
50         icon = 0;
51         this->margin = margin >= 0 ? margin :
52                 BC_WindowBase::get_resources()->popupmenu_margin;
53         this->use_title = use_title;
54         strcpy(this->text, text);
55         for( int i=0; i<TOTAL_IMAGES; ++i ) images[i] = 0;
56         this->data = data;
57         this->w_argument = w;
58         status = BUTTON_UP;
59         pending = 0;
60 }
61
62 BC_PopupMenu::BC_PopupMenu(int x, int y, const char *text,
63                 int use_title, VFrame **data)
64  : BC_SubWindow(x, y, 0, -1, -1)
65 {
66         highlighted = popup_down = 0;
67         menu_popup = 0;
68         icon = 0;
69         this->margin = BC_WindowBase::get_resources()->popupmenu_margin;
70         this->use_title = use_title;
71         strcpy(this->text, text);
72         for( int i=0; i<TOTAL_IMAGES; ++i ) images[i] = 0;
73         this->data = data;
74         this->w_argument = -1;
75         status = BUTTON_UP;
76         pending = 0;
77 }
78
79 BC_PopupMenu::~BC_PopupMenu()
80 {
81         use_title = 0;
82         deactivate();
83         delete menu_popup;
84         for( int i=0; i<TOTAL_IMAGES; ++i ) delete images[i];
85 }
86
87 char* BC_PopupMenu::get_text()
88 {
89         return text;
90 }
91
92 void BC_PopupMenu::set_text(const char *text)
93 {
94         if( use_title ) {
95                 strcpy(this->text, text);
96                 draw_title(1);
97         }
98 }
99
100 void BC_PopupMenu::set_icon(BC_Pixmap *icon)
101 {
102         if( use_title ) {
103                 this->icon = icon;
104                 if( menu_popup ) draw_title(1);
105         }
106 }
107
108 int BC_PopupMenu::initialize()
109 {
110         if( use_title ) {
111                 if( data )
112                         set_images(data);
113                 else
114                 if( BC_WindowBase::get_resources()->popupmenu_images )
115                         set_images(BC_WindowBase::get_resources()->popupmenu_images);
116                 else
117                         set_images(BC_WindowBase::get_resources()->generic_button_images);
118         }
119         else {
120 // Move outside window if no title
121                 x = -10;  y = -10;
122                 w = 10;   h = 10;
123         }
124
125         BC_SubWindow::initialize();
126
127         menu_popup = new BC_MenuPopup;
128         menu_popup->initialize(top_level, 0, 0, 0, this);
129
130         if( use_title ) draw_title(0);
131
132         return 0;
133 }
134
135 int BC_PopupMenu::set_images(VFrame **data)
136 {
137         for( int i=0; i<3; ++i ) {
138                 delete images[i];
139                 images[i] = new BC_Pixmap(parent_window, data[i], PIXMAP_ALPHA);
140         }
141
142         w = w_argument > 0 ? w_argument :
143                 calculate_w(margin, get_text_width(MEDIUMFONT, text), use_title);
144         h = images[BUTTON_UP]->get_h();
145         return 0;
146 }
147
148 int BC_PopupMenu::calculate_w(int margin, int text_width, int use_title)
149 {
150         BC_Resources *resources = get_resources();
151         int l = margin >= 0 ? margin : resources->popupmenu_margin;
152         int r = use_title < 0 ? l : resources->popupmenu_triangle_margin;
153         return l + text_width + r;
154 }
155
156 int BC_PopupMenu::calculate_w(int text_width)
157 {
158         return calculate_w(-1, text_width, 0);
159 }
160
161 int BC_PopupMenu::calculate_h(VFrame **data)
162 {
163         if( !data ) data = BC_WindowBase::get_resources()->popupmenu_images ?
164                 BC_WindowBase::get_resources()->popupmenu_images :
165                 BC_WindowBase::get_resources()->generic_button_images ;
166
167         return data[BUTTON_UP]->get_h();
168 }
169
170 int BC_PopupMenu::add_item(BC_MenuItem *item)
171 {
172         menu_popup->add_item(item);
173         return 0;
174 }
175
176 int BC_PopupMenu::remove_item(BC_MenuItem *item)
177 {
178         menu_popup->remove_item(item);
179         return 0;
180 }
181
182 int BC_PopupMenu::del_item(BC_MenuItem *item)
183 {
184         menu_popup->del_item(item);
185         return 0;
186 }
187
188 int BC_PopupMenu::total_items()
189 {
190         return menu_popup->total_items();
191 }
192
193 BC_MenuItem* BC_PopupMenu::get_item(int i)
194 {
195         return menu_popup->menu_items.values[i];
196 }
197
198 int BC_PopupMenu::get_margin()
199 {
200         return margin;
201 }
202
203 int BC_PopupMenu::draw_face(int dx, int color)
204 {
205         if( !use_title ) return 0;
206
207 // Background
208         draw_top_background(parent_window, 0, 0, w, h);
209         draw_3segmenth(0, 0, w, images[status]);
210
211 // Overlay text
212         if( color < 0 ) color = get_resources()->popup_title_text;
213         set_color(color);
214
215         int offset = status == BUTTON_DN ? 1 : 0;
216         int available_w = get_w() - calculate_w(margin, 0, use_title);
217
218         if( !icon ) {
219                 char truncated[BCTEXTLEN];
220                 truncate_text(truncated, text, available_w);
221                 set_font(MEDIUMFONT);
222                 BC_WindowBase::draw_center_text(
223                         dx + available_w/2 + margin + offset,
224                         (int)((float)get_h()/2 + get_text_ascent(MEDIUMFONT)/2 - 2) + offset,
225                         truncated);
226         }
227
228         if( icon ) {
229                 draw_pixmap(icon,
230                         available_w/ 2 + margin + offset - icon->get_w()/2 ,
231                         get_h()/2 - icon->get_h()/2 + offset);
232         }
233
234         if( use_title >= 0 )
235                 draw_triangle_down_flat(available_w + margin,
236                         get_h()/2 - TRIANGLE_H/2, TRIANGLE_W, TRIANGLE_H);
237         return 1;
238 }
239
240 int BC_PopupMenu::draw_title(int flush)
241 {
242         draw_face(0, -1);
243         flash(flush);
244         return 0;
245 }
246
247 int BC_PopupMenu::deactivate()
248 {
249         if( popup_down ) {
250                 top_level->active_popup_menu = 0;
251                 popup_down = 0;
252                 menu_popup->deactivate_menu();
253
254                 if( use_title ) draw_title(1);    // draw the title
255         }
256         return 0;
257 }
258
259 int BC_PopupMenu::activate_menu()
260 {
261         if( !get_button_down() || !BC_WindowBase::get_resources()->popupmenu_btnup )
262                 return menu_activate();
263         top_level->active_popup_menu = this;
264         pending = 1;
265         return 0;
266 }
267
268 int BC_PopupMenu::menu_activate()
269 {
270         pending = 0;
271         if( !popup_down ) {
272                 int x = this->x;
273                 int y = this->y;
274
275                 top_level->deactivate();
276                 top_level->active_popup_menu = this;
277                 if( !use_title ) {
278                         x = top_level->get_abs_cursor_x(0) - get_w();
279                         y = top_level->get_abs_cursor_y(0) - get_h();
280                         button_press_x = top_level->cursor_x;
281                         button_press_y = top_level->cursor_y;
282                 }
283
284                 if( use_title ) {
285                         Window tempwin;
286                         int new_x, new_y;
287                         XTranslateCoordinates(top_level->display,
288                                 win, top_level->rootwin,
289                                 0, 0, &new_x, &new_y, &tempwin);
290                         menu_popup->activate_menu(new_x, new_y,
291                                 w, h, 0, 1);
292                 }
293                 else
294                         menu_popup->activate_menu(x+3, y+3, w, h, 0, 1);
295                 popup_down = 1;
296                 if( use_title ) draw_title(1);
297         }
298         return 1;
299 }
300
301 int BC_PopupMenu::deactivate_menu()
302 {
303         deactivate();
304         return 0;
305 }
306
307
308 int BC_PopupMenu::reposition_window(int x, int y)
309 {
310         BC_WindowBase::reposition_window(x, y);
311         draw_title(0);
312         return 0;
313 }
314
315 int BC_PopupMenu::focus_out_event()
316 {
317         if( popup_down && !get_button_down() &&
318             !cursor_inside() && !menu_popup->cursor_inside() )
319                 deactivate();
320         return 0;
321 }
322
323
324 int BC_PopupMenu::repeat_event(int64_t duration)
325 {
326         if( status == BUTTON_HI &&
327                 tooltip_text && tooltip_text[0] != 0 &&
328                 duration == top_level->get_resources()->tooltip_delay ) {
329                 show_tooltip();
330                 return 1;
331         }
332         return 0;
333 }
334
335 int BC_PopupMenu::button_press_event()
336 {
337         int result = 0;
338         if( get_buttonpress() == 1 &&
339                 is_event_win() &&
340                 use_title ) {
341                 top_level->hide_tooltip();
342                 if( status == BUTTON_HI || status == BUTTON_UP ) status = BUTTON_DN;
343                 activate_menu();
344                 draw_title(1);
345                 return 1;
346         }
347
348         // Scrolling section
349         if( is_event_win()
350                 && (get_buttonpress() == 4 || get_buttonpress() == 5)
351                 && menu_popup->total_items() > 1  ) {
352                 int theval = -1;
353                 for( int i=0; i<menu_popup->total_items(); ++i ) {
354                         if( !strcmp(menu_popup->menu_items.values[i]->get_text(),get_text()) ) {
355                                 theval = i;
356                                 break;
357                         }
358                 }
359
360                 if( theval == -1 ) theval = 0;
361                 else if( get_buttonpress() == 4 ) --theval;
362                 else if( get_buttonpress() == 5 ) ++theval;
363
364                 if( theval < 0 )
365                         theval = 0;
366                 else if( theval >= menu_popup->total_items() )
367                         theval = menu_popup->total_items() - 1;
368
369                 BC_MenuItem *tmp = menu_popup->menu_items.values[theval];
370                 set_text(tmp->get_text());
371                 result = tmp->handle_event();
372                 if( !result )
373                         result = this->handle_event();
374         }
375         if( popup_down ) {
376 // Menu is down so dispatch to popup.
377                 menu_popup->dispatch_button_press();
378                 result = 1;
379         }
380
381         return result;
382 }
383
384 int BC_PopupMenu::button_release_event()
385 {
386 // try the title
387         int result = 0;
388
389         if( is_event_win() && use_title ) {
390                 hide_tooltip();
391                 if( status == BUTTON_DN ) {
392                         status = BUTTON_HI;
393                         draw_title(1);
394                 }
395         }
396
397         if( pending )
398                 return menu_activate();
399
400         if( !use_title && status == BUTTON_DN ) {
401                 result = 1;
402         }
403         else if( popup_down && menu_popup->cursor_inside() ) {
404 // Menu is down so dispatch to popup.
405                 result = menu_popup->dispatch_button_release();
406         }
407 // released outside popup
408         if( get_resources()->popupmenu_btnup && !result && popup_down ) {
409                 deactivate();
410                 result = 1;
411         }
412         hide_tooltip();
413
414         return result;
415 }
416
417 int BC_PopupMenu::translation_event()
418 {
419 //printf("BC_PopupMenu::translation_event 1\n");
420         if( popup_down ) menu_popup->dispatch_translation_event();
421         return 0;
422 }
423
424 int BC_PopupMenu::cursor_leave_event()
425 {
426
427         if( status == BUTTON_HI && use_title ) {
428                 status = BUTTON_UP;
429                 draw_title(1);
430                 hide_tooltip();
431         }
432
433 // dispatch to popup
434         if( popup_down ) {
435                 if( !get_button_down() && !menu_popup->cursor_inside() ) {
436                         status = BUTTON_UP;
437 //                      deactivate_menu();
438                 }
439                 menu_popup->dispatch_cursor_leave();
440         }
441
442         return 0;
443 }
444
445
446 int BC_PopupMenu::cursor_enter_event()
447 {
448         if( is_event_win() && use_title ) {
449                 if( top_level->button_down ) {
450                         status = BUTTON_DN;
451                 }
452                 else
453                 if( status == BUTTON_UP )
454                         status = BUTTON_HI;
455                 draw_title(1);
456         }
457
458         return 0;
459 }
460
461 int BC_PopupMenu::cursor_motion_event()
462 {
463         int result = 0;
464
465 // This menu is down.
466         if( popup_down ) {
467                 result = menu_popup->dispatch_motion_event();
468         }
469
470         if( !result && use_title && is_event_win() ) {
471                 if( highlighted ) {
472                         if( !cursor_inside() ) {
473                                 highlighted = 0;
474                                 draw_title(1);
475                         }
476                 }
477                 else {
478                         if( cursor_inside() ) {
479                                 highlighted = 1;
480                                 draw_title(1);
481                                 result = 1;
482                         }
483                 }
484         }
485
486         return result;
487 }
488
489 int BC_PopupMenu::drag_start_event()
490 {
491 //printf("BC_PopupMenu::drag_start_event %d\n", popup_down);
492         if( popup_down ) return 1;
493         return 0;
494 }
495
496 int BC_PopupMenu::drag_stop_event()
497 {
498         if( popup_down ) return 1;
499         return 0;
500 }
501
502 int BC_PopupMenu::drag_motion_event()
503 {
504         if( popup_down ) return 1;
505         return 0;
506 }
507