]> git.lyx.org Git - lyx.git/blob - src/frontends/gnome/Menubar_pimpl.C
Small fix from Angus
[lyx.git] / src / frontends / gnome / Menubar_pimpl.C
1 // -*- C++ -*-
2 /* This file is part of
3  * ====================================================== 
4  * 
5  *           LyX, The Document Processor
6  *       
7  *          Copyright 2000 The LyX Team.
8  *
9  * ====================================================== */
10
11 #ifdef __GNUG__
12 #pragma implementation
13 #endif
14
15 #include <config.h>
16
17 #include <algorithm>
18 #include <cctype>
19 #include "support/lstrings.h"
20 #include "support/filetools.h"
21 #include "support/StrPool.h"
22 #include "support/LAssert.h"
23 #include "debug.h"
24 #include "LyXAction.h"
25 #include "lyxfunc.h"
26 #include "kbmap.h"
27 #include "bufferlist.h"
28 #include "lastfiles.h"
29 #include "LyXView.h"
30 #include "MenuBackend.h"
31 #include "Menubar_pimpl.h"
32 #include "lyxtext.h"
33 #include "exporter.h"
34
35 #include "mainapp.h"
36
37 #include <gtk--/menu.h>
38
39 using SigC::slot;
40 using SigC::bind;
41
42 using std::endl;
43
44 // temporary solution for LyXView
45 extern GLyxAppWin * mainAppWin;
46
47 // Some constants
48 extern kb_keymap * toplevel_keymap;
49 extern LyXAction lyxaction;
50 extern BufferList bufferlist;
51 extern LastFiles * lastfiles; 
52
53
54
55 Menubar::Pimpl::Pimpl(LyXView * view, MenuBackend const & mb) 
56   : owner_(view), menubackend_(&mb), ignore_action_(false)
57 {
58   
59 }
60
61 Menubar::Pimpl::~Pimpl() 
62 {
63   if (utoc_.connected()) utoc_.disconnect();
64 }
65
66 void Menubar::Pimpl::set(string const & menu_name) 
67 {
68   // if (current_menu_name_ != menu_name)  // disabled until Lastfiles and Documents are added dynamically to menu
69     {
70       current_menu_name_ = menu_name;
71
72       // clean up the lists
73       toc_.clear();
74       if (utoc_.connected()) utoc_.disconnect();
75       
76       // compose new menu
77       vector<Gnome::UI::Info> menus;
78       composeUIInfo(current_menu_name_, menus, "");
79
80       // set menu
81       Menu_ = menus;
82       mainAppWin->set_menu(Menu_);
83
84       // connect all menu items to correspoding action
85       wid_act_.clear();
86       ignore_action_ = true;
87       connectWidgetToAction(Menu_.gtkobj());
88       ignore_action_ = false;
89
90       // update state of the items
91       update();
92       updateAllLists();
93     }
94 }
95
96 void Menubar::Pimpl::updateAllLists()
97 {
98   // update lists
99   if (toc_.size() > 0)
100     {
101       vector<Buffer::TocItem> toclist = (owner_->view()->buffer()->getTocList())[Buffer::TOC_TOC];
102       updateList(&toclist, &toc_);
103     }
104 }
105
106 int const max_number_of_items = 25;
107 void Menubar::Pimpl::updateList(vector<Buffer::TocItem> * toclist, vector<ListsHolder> * pgui) 
108 {
109   vector<ListsHolder> & gui = *pgui;
110   int szGui = gui.size();
111   int i;
112   for (i=0; i < szGui; ++i)
113     {
114       int oldsz = gui[i].lst.size();
115       vector<Gnome::UI::Info> menu;
116       string label;
117
118       menu.push_back(Gnome::UI::Item(Gnome::UI::Icon(GNOME_STOCK_MENU_REFRESH),
119                                      _("Refresh"), slot(this, &Menubar::Pimpl::updateAllLists)));
120
121       if (toclist->size() > max_number_of_items)
122         composeTocUIInfo(menu, *toclist, toclist->begin(), 0);
123       else
124         {
125           vector<Buffer::TocItem>::const_iterator end = toclist->end();
126           for (vector<Buffer::TocItem>::const_iterator it = toclist->begin();
127                it != end; ++it)
128             
129             {
130               label = string(4*(*it).depth,' ')+(*it).str;
131               
132               menu.push_back(Gnome::UI::Item(label,
133                                              bind<Buffer::TocItem>(slot(this, &Menubar::Pimpl::callbackToc), (*it)),
134                                              label));
135             }
136         }
137       
138       gui[i].lst = menu;
139       mainAppWin->update_menu(gui[i].path, oldsz, gui[i].lst);
140     }
141 }
142
143 vector<Buffer::TocItem>::const_iterator
144 Menubar::Pimpl::composeTocUIInfo(vector<Gnome::UI::Info> & menu,
145                                  vector<Buffer::TocItem> const & toclist,
146                                  vector<Buffer::TocItem>::const_iterator begin,
147                                  int mylevel)
148 {
149   string label = _("<No Name>");
150
151   vector<Buffer::TocItem>::const_iterator end = toclist.end();
152   vector<Buffer::TocItem>::const_iterator it;
153   for (it = begin; it != end && (*it).depth >= mylevel; ++it)
154     {
155       if ( (*it).depth == mylevel &&
156            (it+1 == end || (*(it+1)).depth <= mylevel) )
157         {
158           label = (*it).str;
159           menu.push_back(Gnome::UI::Item(label,
160                                        bind<Buffer::TocItem>(slot(this, &Menubar::Pimpl::callbackToc), (*it)),
161                                          label));
162         }
163       else
164         {
165           vector<Gnome::UI::Info> submenu;
166           if ( (*it).depth == mylevel )
167             {
168               label = (*it).str;
169               submenu.push_back(Gnome::UI::Item(label,
170                                                 bind<Buffer::TocItem>(slot(this, &Menubar::Pimpl::callbackToc), (*it)),
171                                                 label));
172               ++it;    
173             }
174           it = composeTocUIInfo(submenu, toclist, it, mylevel+1);
175           menu.push_back(Gnome::UI::Menu(label,submenu,label));
176         }
177     }
178   --it;
179   return it;
180 }
181
182 void Menubar::Pimpl::callback(int action)
183 {
184   // Dispatch action OR record action to local variable (see connectWidgetToAction)
185   if (!ignore_action_) {
186       Pimpl::update();
187       owner_->getLyXFunc()->Dispatch(action);
188   } else
189       action_ = action;
190 }
191
192 void Menubar::Pimpl::callbackToc(Buffer::TocItem tg)
193 {
194   if (!owner_->view()->available()) return;
195   
196   owner_->view()->beforeChange();
197   owner_->view()->text->SetCursor( owner_->view(), tg.par, 0 );
198   owner_->view()->text->sel_cursor = owner_->view()->text->cursor;
199   owner_->view()->update(BufferView::SELECT|BufferView::FITCUR);
200 }
201
202 void Menubar::Pimpl::composeUIInfo(string const & menu_name, vector<Gnome::UI::Info> & Menus, string rootpath)
203 {
204   string path = rootpath;
205     
206   if (!menubackend_->hasMenu(menu_name))
207     {
208       cout << "ERROR:composeUIInfo: Unknown menu `" << menu_name
209            << "'" << endl;
210       return;
211     }
212
213   Menu menu = Menu();
214   menubackend_->getMenu(menu_name).expand(menu, owner_->buffer());
215
216   for (Menu::const_iterator i = menu.begin(); i != menu.end(); ++i)
217     {
218       MenuItem item = (*i);
219       switch(item.kind()) {
220
221       case MenuItem::Command: {
222         string label = item.label();
223
224         path = rootpath + label;
225         
226         if (label.find(item.shortcut()) != string::npos)
227           label.insert(label.find(item.shortcut()), "_");
228
229         LyXFunc::func_status flag = owner_->getLyXFunc()->getStatus(item.action());
230
231         Gnome::UI::Info gitem;
232         SigC::Slot0<void> cback = bind<int>(slot(this, &Menubar::Pimpl::callback),item.action());
233
234         {
235           using namespace Gnome::MenuItems;
236           int ac = item.action();
237           kb_action action;
238           string argument;
239           if (lyxaction.isPseudoAction(ac))
240             action = lyxaction.retrieveActionArg(ac, argument);
241           else
242             action = static_cast<kb_action>(ac);
243
244           switch(action) {
245           case LFUN_MENUOPEN:
246             gitem = Open(cback);
247             break;
248           case LFUN_QUIT:
249             gitem = Exit(cback);
250             break;
251           case LFUN_CLOSEBUFFER:
252             gitem = Close(cback);
253             break;
254           case LFUN_MENUWRITE:
255             gitem = Save(cback);
256             break;
257           case LFUN_MENUWRITEAS:
258             gitem = SaveAs(cback);
259             break;
260           case LFUN_BUFFER_PRINT:
261             gitem = Print(cback);
262             break;
263           case LFUN_CUT:
264             gitem = Cut(cback);
265             break;
266           case LFUN_COPY:
267             gitem = Copy(cback);
268             break;
269           case LFUN_PASTE:
270             gitem = Paste(cback);
271             break;
272           case LFUN_UNDO:
273             gitem = Gnome::MenuItems::Undo(cback); // confused with class Undo
274             break;
275           case LFUN_REDO:
276             gitem = Redo(cback);
277             break;
278           case LFUN_DIALOG_PREFERENCES:
279             gitem = Preferences(cback);
280             break;
281           case LFUN_MENUNEW:
282             gitem = Gnome::UI::Item(Gnome::UI::Icon(GNOME_STOCK_MENU_NEW),
283                                     label, cback, lyxaction.helpText(item.action()));
284             break;
285           case LFUN_MENUNEWTMPLT:
286             gitem = Gnome::UI::Item(Gnome::UI::Icon(GNOME_STOCK_MENU_NEW), 
287                                     label, cback, lyxaction.helpText(item.action()));
288             break;
289           case LFUN_MENUSEARCH:
290             gitem = Gnome::UI::Item(Gnome::UI::Icon(GNOME_STOCK_MENU_SRCHRPL), 
291                                     label, cback, lyxaction.helpText(item.action()));
292             break;
293           case LFUN_SPELLCHECK:
294             gitem = Gnome::UI::Item(Gnome::UI::Icon(GNOME_STOCK_MENU_SPELLCHECK), 
295                                     label, cback, lyxaction.helpText(item.action()));
296             break;
297           default:
298             gitem = Gnome::UI::Item(label, cback, lyxaction.helpText(item.action()));
299             break;
300           }
301         }
302
303         // first handle optional entries.
304         if (item.optional() && (flag & LyXFunc::Disabled)) {
305             lyxerr[Debug::GUI] 
306                 << "Skipping optional item " << item.label() << endl; 
307             break;
308         }
309         if ((flag & LyXFunc::ToggleOn) || (flag & LyXFunc::ToggleOff))
310           gitem = Gnome::UI::ToggleItem(label, cback, lyxaction.helpText(item.action()));
311
312         Menus.push_back(gitem);
313         break;
314       }
315       
316       case MenuItem::Submenu: {
317         vector<Gnome::UI::Info> submenu;
318         string label = item.label();
319
320         path = rootpath + label;
321         
322         if (label.find(item.shortcut()) != string::npos)
323           label.insert(label.find(item.shortcut()), "_");
324         composeUIInfo(item.submenu(), submenu, path + "/");
325         Menus.push_back(Gnome::UI::Menu(label,submenu,label));
326         break;
327       }
328
329       case MenuItem::Separator: {
330
331         path = rootpath + "<Separator>";
332         
333         Menus.push_back(Gnome::UI::Separator());
334         break;
335       }
336
337       case MenuItem::Toc: {
338         ListsHolder t;
339         t.path = path;
340         toc_.push_back(t);
341         break;
342       }
343       
344       case MenuItem::Documents: 
345       case MenuItem::Lastfiles: 
346       case MenuItem::ViewFormats:
347       case MenuItem::UpdateFormats:
348       case MenuItem::ExportFormats:
349                         lyxerr << "Menubar::Pimpl::create_submenu: "
350                           "this should not happen" << endl;
351                         break;
352       }
353     }
354 }
355
356 void Menubar::Pimpl::connectWidgetToAction(GnomeUIInfo * guinfo)
357 {
358   for (; guinfo->type !=  GnomeUIInfoType(GNOME_APP_UI_ENDOFINFO); ++guinfo)
359     {
360       if ( ( guinfo->type == GnomeUIInfoType(GNOME_APP_UI_ITEM) ||
361              guinfo->type == GnomeUIInfoType(GNOME_APP_UI_TOGGLEITEM) ) &&
362            guinfo->moreinfo != NULL )
363         {
364           (*((void(*)(void *, void *))(guinfo->moreinfo)))(NULL, guinfo->user_data);
365           wid_act_.push_back( GtkWidgetToAction( guinfo->widget, action_ ) );
366         }
367       else if ( guinfo->type == GnomeUIInfoType(GNOME_APP_UI_SUBTREE) ||
368                 guinfo->type == GnomeUIInfoType(GNOME_APP_UI_RADIOITEMS) )
369         {
370           connectWidgetToAction(  (GnomeUIInfo *)(guinfo->moreinfo) );
371         }
372     }
373 }
374
375 void Menubar::Pimpl::update()
376 {
377   vector<GtkWidgetToAction>::const_iterator end=wid_act_.end();
378   for (vector<GtkWidgetToAction>::const_iterator i = wid_act_.begin(); i != end; ++i)
379     {
380       GtkWidgetToAction wa = (*i);
381       LyXFunc::func_status flag = owner_->getLyXFunc()->getStatus(wa.action_);
382
383       if ( flag & (LyXFunc::Disabled | LyXFunc::Unknown) ) gtk_widget_set_sensitive(wa.widget_, false);
384       else gtk_widget_set_sensitive(wa.widget_, true);
385
386       if ( flag & LyXFunc::ToggleOn )
387         {
388           ignore_action_=true;
389           gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(wa.widget_), true);
390           ignore_action_=false;
391         }
392
393       if ( flag & LyXFunc::ToggleOff )
394         {
395           ignore_action_=true;
396           gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(wa.widget_), false);
397           ignore_action_=false;
398         }
399     }
400 }
401
402 void Menubar::Pimpl::openByName(string const &)
403 {
404 //    Pimpl::update();
405 }