]> git.lyx.org Git - lyx.git/blob - src/mathed/math_panel.C
first go at mathed file cleanup
[lyx.git] / src / mathed / math_panel.C
1 /*
2  *  File:        math_panel.C
3  *  Purpose:     Mathed GUI for lyx
4  *  Author:      Alejandro Aguilar Sierra <asierra@servidor.unam.mx>
5  *  Created:     March 28, 1996
6  * 
7  *  Dependencies: Xlib, Xpm, XForms, Lyx
8  *
9  *  Copyright: 1996, Alejandro Aguilar Sierra 
10  *
11  *   You are free to use and modify it under the terms of
12  *   the GNU General Public Licence version 2 or later.
13  */
14
15 #include <config.h>
16
17 #include "Lsstream.h"
18
19 #include FORMS_H_LOCATION
20
21 #include "lyx_gui_misc.h" 
22 #include "math_panel.h"
23 #include "symbol_def.h"
24 #include "formula.h"
25 #include "lyxfunc.h"
26 #include "gettext.h"
27
28 /* Bitmaps */
29 #include "delim.xbm"
30 #include "delim0.xpm"
31 #include "delim.xpm"
32 #include "deco.xbm"
33 #include "deco.xpm"
34 #include "space.xpm"
35 #include "sqrt.xpm"
36 #include "frac.xpm"
37 #include "matrix.xpm"
38 #include "equation.xpm"
39
40 #if 0
41 using std::free;
42 #endif
43
44 static LyXFunc * lyxfunc = 0;
45
46 //static FD_panel* symb_form= 0;
47
48 FD_panel  * fd_panel;
49 FD_delim  * fd_delim;
50 FD_deco   * fd_deco;
51 FD_space  * fd_space;
52 FD_matrix * fd_matrix;
53
54 int delim_code[] = {   
55    '(', ')', LM_lceil,  LM_rceil,  LM_uparrow,  LM_Uparrow,
56    '[', ']', LM_lfloor,  LM_rfloor,  LM_updownarrow, LM_Updownarrow,
57    '{', '}',  '/', LM_backslash,  LM_downarrow,  LM_Downarrow,
58    LM_langle,  LM_rangle, '|', LM_Vert, '.', 0
59 };
60
61 // indexes to get the left and right versions of each delimiter
62 // Contributed by Pablo De Napoli (pdenapo@dm.uba.ar)
63 int delim_lversion[] = { 0,0,2,2,4,5,
64                          6,6,8,8,10,11,
65                          12,12,14,15,16,17,
66                          18,18,20,21,22,23 };
67
68 int delim_rversion[] = { 1,1,3,3,4,5,
69                         7,7,9,9,10,11,
70                         13,13,14,15,16,17,
71                         19,19,20,21,22,23 };
72
73
74 static char const * deco_code[] = {
75    "widehat", "widetilde", "overbrace", "overleftarrow", "overrightarrow", 
76    "overline", "underbrace", "underline"
77 };
78
79
80 static char const * func_code[] = {
81     "arccos", "arcsin", "arctan", "arg", "bmod",
82     "cos", "cosh", "cot", "coth", "csc", "deg",
83     "det", "dim", "exp", "gcd", "hom", "inf", "ker",
84     "lg", "lim", "liminf", "limsup", "ln", "log",
85     "max", "min", "sec", "sin", "sinh", "sup",
86     "tan", "tanh"
87 };
88
89 static char h_align_str[80] = "c";
90
91 /* callbacks for form panel */
92 void button_cb(FL_OBJECT * ob, long data)
93 {   
94    extern void free_symbols_form();
95    switch (data)  {
96     case MM_GREEK:
97     case MM_VARSIZE:
98     case MM_BRELATS:
99     case MM_ARROW:
100     case MM_BOP:
101     case MM_MISC: 
102       {    
103          BitmapMenu * menu = static_cast<BitmapMenu *>(ob->u_vdata);
104          menu->Show();  
105          break;
106       }
107     case MM_FRAC:
108        lyxfunc->Dispatch(LFUN_INSERT_MATH, "frac");
109       break;
110     case MM_SQRT:
111        lyxfunc->Dispatch(LFUN_INSERT_MATH, "sqrt");
112       break;
113     case MM_DELIM:
114       fl_show_form(fd_delim->delim,
115                    FL_PLACE_MOUSE | FL_FREE_SIZE, FL_TRANSIENT,
116                    _("Delimiter"));
117        fl_set_form_atclose(fd_delim->delim, CancelCloseBoxCB, 0);
118       break;
119     case MM_DECO:
120       fl_show_form(fd_deco->deco,
121                    FL_PLACE_MOUSE | FL_FREE_SIZE, FL_TRANSIENT,
122                    _("Decoration"));
123        fl_set_form_atclose(fd_deco->deco, CancelCloseBoxCB, 0);
124       break;
125     case MM_SPACE:
126       fl_show_form(fd_space->space,
127                    FL_PLACE_MOUSE | FL_FREE_SIZE, FL_TRANSIENT,
128                    _("Spacing"));
129        fl_set_form_atclose(fd_space->space, CancelCloseBoxCB, 0);
130       break;
131     case MM_MATRIX:
132       fl_show_form(fd_matrix->matrix,
133                    FL_PLACE_MOUSE | FL_FREE_SIZE, FL_TRANSIENT,
134                    _("Matrix"));
135        fl_set_form_atclose(fd_matrix->matrix, CancelCloseBoxCB, 0);
136       break;
137     case MM_EQU:
138        lyxfunc->Dispatch(LFUN_MATH_DISPLAY);
139       break;
140     case MM_FUNC:
141       {
142           int i = fl_get_browser(fd_panel->func_browse) - 1;
143           lyxfunc->Dispatch(LFUN_INSERT_MATH, func_code[i]);
144           break;
145       }
146     case 100:
147       free_symbols_form();
148       break;
149    }
150 }
151
152
153 /* callbacks for form delim */
154 void delim_cb(FL_OBJECT *, long data)
155 {
156    int left = fd_delim->left->u_ldata, right= fd_delim->right->u_ldata;
157    int side = (fl_get_button(fd_delim->right)!= 0);
158    
159    switch (data) {
160     case MM_APPLY:
161     case MM_OK:
162       {
163               std::ostringstream ost;
164               ost << delim_code[left] << ' ' << delim_code[right];
165               lyxfunc->Dispatch(LFUN_MATH_DELIM, ost.str().c_str());
166               if (data == MM_APPLY) break;
167       }
168     case MM_CLOSE: fl_hide_form(fd_delim->delim); break;
169     case 2: 
170       {
171           int i = fl_get_bmtable(fd_delim->menu);
172           int button = fl_get_bmtable_numb(fd_delim->menu);
173           bool both = (button==FL_MIDDLE_MOUSE);
174           
175           if (i>= 0) {
176
177               if (side || (button== FL_RIGHT_MOUSE)) {
178                   right = i;
179               } else {
180                   left = i;
181                   if (both)
182                     right = delim_rversion[i];
183               }   
184           }
185           Pixmap p1, p2;
186           p1 = fl_get_pixmap_pixmap(fd_delim->pix, &p1, &p2);
187           fl_draw_bmtable_item(fd_delim->menu, left, p1, 0, 0);
188           fl_draw_bmtable_item(fd_delim->menu, right, p1, 16, 0);
189           fl_redraw_object(fd_delim->pix);
190           
191           fd_delim->left->u_ldata = left;
192           fd_delim->right->u_ldata = right;
193
194           break;
195       }
196     case 3: break;
197     case 4: break;
198    }
199 }
200
201 /* callbacks for form matrix */
202 void matrix_cb(FL_OBJECT *, long data)
203 {
204    static char v_align_c[] = "tcb";
205  
206    switch (data) {
207     case MM_APPLY:
208     case MM_OK: 
209       {
210          char c = v_align_c[fl_get_choice(fd_matrix->valign) - 1];
211          char const * sh = fl_get_input(fd_matrix->halign);
212          int nx = int(fl_get_slider_value(fd_matrix->columns) + 0.5);
213          int ny = int(fl_get_slider_value(fd_matrix->rows) + 0.5);
214          if (data == MM_OK) fl_hide_form(fd_matrix->matrix);
215          std::ostringstream ost;
216          ost << nx << ' ' << ny << ' ' << c << sh;
217          lyxfunc->Dispatch(LFUN_INSERT_MATRIX, ost.str().c_str());
218          break;
219       }
220     case MM_CLOSE: fl_hide_form(fd_matrix->matrix); break;
221     case 2: 
222       {
223          int nx = int(fl_get_slider_value(fd_matrix->columns)+0.5);
224          for (int i = 0; i < nx; ++i) h_align_str[i] = 'c';
225          //memset(h_align_str, 'c', nx);
226          h_align_str[nx] = '\0';
227 //       fl_freeze_form(fd_form_main->form_main);
228 //      fl_addto_form(fd_form_main->form_main);
229
230          fl_set_input(fd_matrix->halign, h_align_str);  
231          fl_redraw_object(fd_matrix->halign);    
232          break;
233       }
234    }
235 }
236
237 /* callbacks for form deco */
238 void deco_cb(FL_OBJECT *, long data)
239 {
240    switch (data) {
241     case MM_APPLY:
242     case MM_OK:
243       { 
244          int i = fl_get_bmtable(fd_deco->menu);
245          // ideally the callback should not be called if the index is
246          // greater than the maxitem of the bmtable, but I do not know
247          // how to enforce that (JMarc)
248          if (i <= sizeof(deco_code)/sizeof(deco_code[0])) {
249            lyxfunc->Dispatch(LFUN_INSERT_MATH, deco_code[i]);
250            if (data == MM_APPLY) break;
251          }
252          else break;
253       }
254     case MM_CLOSE: fl_hide_form(fd_deco->deco); break;
255    }
256 }
257
258
259 /* callbacks for form space */
260 void space_cb(FL_OBJECT *, long data)
261 {
262    static short sp = -1;
263    extern char * latex_mathspace[];
264    
265    if (data >= 0 && data < 6) 
266       sp = short(data);
267    else
268    switch (data) {
269     case MM_APPLY:
270     case MM_OK:
271       { 
272           if (sp>= 0) 
273             lyxfunc->Dispatch(LFUN_INSERT_MATH, latex_mathspace[sp]);
274          if (data == MM_APPLY) break;
275       }
276     case MM_CLOSE: fl_hide_form(fd_space->space); break;
277    }
278 }
279
280
281 extern "C" int align_filter(FL_OBJECT *, char const *, char const * cur, int c)
282 {
283    int n = int(fl_get_slider_value(fd_matrix->columns)+0.5) - strlen(cur);
284    return ((c == 'c'||c == 'l'||c == 'r') && n>= 0) ? FL_VALID: FL_INVALID;
285 }
286
287
288 char const ** mathed_get_pixmap_from_icon(int d)
289 {
290    switch (d) {
291     case MM_FRAC: return frac;
292     case MM_SQRT: return sqrt_xpm;
293     case MM_DELIM: return delim;
294     case MM_MATRIX: return matrix;
295     case MM_EQU: return equation; 
296     case MM_DECO: return deco; 
297     case MM_SPACE: return space_xpm; 
298     default: return 0;
299    }
300 }
301
302
303 FD_panel * create_math_panel( )
304 {
305    fd_panel = create_form_panel();
306    fd_delim = create_form_delim();
307    fd_deco = create_form_deco();
308    fd_space = create_form_space();
309    fd_matrix = create_form_matrix();
310
311    /* fill-in form initialization code */
312    fl_set_button(fd_delim->left, 1);
313    fl_set_pixmap_data(fd_delim->pix, const_cast<char**>(delim0));
314    fl_set_bmtable_data(fd_delim->menu, 6, 4, delim_width, delim_height,
315                        delim_bits);
316    fl_set_bmtable_maxitems(fd_delim->menu, 23);
317    
318    fl_set_pixmap_data(fd_panel->sqrt, const_cast<char**>(sqrt_xpm));
319    fl_set_pixmap_data(fd_panel->frac, const_cast<char**>(frac));
320    fl_set_pixmap_data(fd_panel->delim, const_cast<char**>(delim));
321    fl_set_pixmap_data(fd_panel->deco, const_cast<char**>(deco));
322    fl_set_pixmap_data(fd_panel->space, const_cast<char**>(space_xpm));
323    fl_set_pixmap_data(fd_panel->matrix, const_cast<char**>(matrix));
324    fl_set_pixmap_data(fd_panel->equation, const_cast<char**>(equation));
325
326    for (int i = 0; i < 32; ++i) {
327        fl_add_browser_line(fd_panel->func_browse, func_code[i]);
328    }
329     
330    fl_addto_choice(fd_matrix->valign, _("Top | Center | Bottom"));
331    fl_set_choice(fd_matrix->valign, 2);
332    fl_set_input(fd_matrix->halign, h_align_str);
333    fl_set_input_filter(fd_matrix->halign, align_filter);
334    
335    fl_set_bmtable_data(fd_deco->menu, 3, 3, deco_width, deco_height,
336                        deco_bits);
337    fl_set_bmtable_maxitems(fd_deco->menu, 8);
338
339    fd_delim->left->u_ldata = 0;
340    fd_delim->right->u_ldata = 1;
341     
342    return fd_panel;
343 }
344
345 extern BitmapMenu * sym_menu;
346 extern void  create_symbol_menues(FD_panel *);
347
348
349 void free_symbols_form()
350 {
351    if (fd_panel) {
352       fl_hide_form(fd_panel->panel);
353       fl_free_form(fd_panel->panel);
354       delete sym_menu;
355       free(fd_panel);
356       fd_panel = 0;  
357    }
358 }
359
360
361 extern "C" int AtClose_symbols_form(FL_FORM *, void *)
362 {
363   free_symbols_form();
364   return FL_IGNORE;
365 }
366
367
368 void show_symbols_form(LyXFunc * lf)
369 {
370     lyxfunc = lf;
371     if (!fd_panel) {
372         fd_panel = create_math_panel();
373         fl_register_raw_callback(fd_panel->panel, 
374                                  ButtonPressMask|KeyPressMask, C_peek_event);
375         create_symbol_menues(fd_panel); 
376         fl_set_form_atclose(fd_panel->panel, AtClose_symbols_form, 0);
377     }
378     if (fd_panel->panel->visible) {
379         fl_raise_form(fd_panel->panel);
380     } else {
381       fl_show_form(fd_panel->panel,
382                    FL_PLACE_MOUSE | FL_FREE_SIZE, FL_TRANSIENT,
383                    _("Math Panel"));
384     }
385 }