]> git.lyx.org Git - lyx.git/blob - src/frontends/qt4/lyx_gui.C
Fix for Qt4 crash on exit under linux by Georg Baum and myself.
[lyx.git] / src / frontends / qt4 / lyx_gui.C
1 /**
2  * \file qt4/lyx_gui.C
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author unknown
7  * \author John Levon
8  *
9  * Full author contact details are available in file CREDITS.
10  */
11
12 #include <config.h>
13
14 #include "lyx_gui.h"
15
16 // FIXME: move this stuff out again
17 #include "bufferlist.h"
18 #include "BufferView.h"
19 #include "Color.h"
20 #include "funcrequest.h"
21 #include "LColor.h"
22 #include "lyx_main.h"
23 #include "LyXAction.h"
24 #include "lyxfunc.h"
25 #include "lyxrc.h"
26 #include "lyxserver.h"
27 #include "lyxsocket.h"
28 #include "session.h"
29
30 #include "graphics/LoaderQueue.h"
31
32 #include "support/lstrings.h"
33 #include "support/convert.h"
34 #include "support/os.h"
35 #include "support/package.h"
36 #include "debug.h"
37
38 // Dear Lord, deliver us from Evil, aka the Qt headers
39 // Qt defines a macro 'signals' that clashes with a boost namespace.
40 // All is well if the namespace is visible first.
41 #include <boost/signal.hpp> // FIXME: Is this needed? (Lgb)
42 #include <boost/bind.hpp>
43 #include <boost/shared_ptr.hpp>
44
45 #include "QtView.h"
46 #include "lcolorcache.h"
47 #include "qfont_loader.h"
48 #include "QLImage.h"
49 #include "qt_helpers.h"
50 #include "socket_callback.h"
51
52 #ifdef Q_WS_MACX
53 #include <Carbon/Carbon.h>
54 #endif
55
56 #include <QApplication>
57 #include <QEventLoop>
58 #include <QTranslator>
59 #include <QTextCodec>
60
61 using lyx::support::ltrim;
62 using lyx::support::package;
63
64 using lyx::frontend::QtView;
65
66 namespace os = lyx::support::os;
67
68 using boost::shared_ptr;
69
70 #ifndef CXX_GLOBAL_CSTD
71 using std::exit;
72 #endif
73
74 using std::map;
75 using std::vector;
76 using std::string;
77
78
79 extern BufferList bufferlist;
80
81 namespace {
82
83 int getDPI()
84 {
85         QWidget w;
86         return int(0.5 * (w.logicalDpiX() + w.logicalDpiY()));
87 }
88
89 map<int, shared_ptr<socket_callback> > socket_callbacks;
90
91 } // namespace anon
92
93 // FIXME: wrong place !
94 LyXServer * lyxserver;
95 LyXServerSocket * lyxsocket;
96
97 // in QLyXKeySym.C
98 extern void initEncodings();
99
100 #ifdef Q_WS_X11
101 extern bool lyxX11EventFilter(XEvent * xev);
102 #endif
103
104 #ifdef Q_WS_MACX
105 extern bool macEventFilter(EventRef event);
106 extern pascal OSErr
107 handleOpenDocuments(const AppleEvent* inEvent, AppleEvent* /*reply*/,
108                     long /*refCon*/);
109 #endif
110
111 class LQApplication : public QApplication
112 {
113 public:
114         LQApplication(int & argc, char ** argv);
115         ~LQApplication();
116 #ifdef Q_WS_X11
117         bool x11EventFilter (XEvent * ev) { return lyxX11EventFilter(ev); }
118 #endif
119 #ifdef Q_WS_MACX
120         bool macEventFilter(EventRef event);
121 #endif
122 };
123
124
125 LQApplication::LQApplication(int & argc, char ** argv)
126         : QApplication(argc, argv)
127 {
128 #ifdef Q_WS_MACX
129         AEInstallEventHandler(kCoreEventClass, kAEOpenDocuments,
130                               NewAEEventHandlerUPP(handleOpenDocuments),
131                               0, false);
132 #endif
133 }
134
135
136 LQApplication::~LQApplication()
137 {}
138
139
140 #ifdef Q_WS_MACX
141 bool LQApplication::macEventFilter(EventRef event)
142 {
143         if (GetEventClass(event) == kEventClassAppleEvent) {
144                 EventRecord eventrec;
145                 ConvertEventRefToEventRecord(event, &eventrec);
146                 AEProcessAppleEvent(&eventrec);
147
148                 return false;
149         }
150         return false;
151 }
152 #endif
153
154
155 namespace {
156
157 LQApplication * app = 0;
158
159 }
160
161
162 namespace lyx_gui {
163
164 bool use_gui = true;
165
166 void parse_init(int & argc, char * argv[])
167 {
168         // Force adding of font path _before_ QApplication is initialized
169         FontLoader::initFontPath();
170
171         app = new LQApplication(argc, argv);
172
173         // install translation file for Qt built-in dialogs
174         // These are only installed since Qt 3.2.x
175         static QTranslator qt_trans(0);
176         if (qt_trans.load(QString("qt_") + QTextCodec::locale(),
177                           qInstallPathTranslations())) {
178                 qApp->installTranslator(&qt_trans);
179                 // even if the language calls for RtL, don't do that
180                 qApp->setReverseLayout(false);
181                 lyxerr[Debug::GUI]
182                         << "Successfully installed Qt translations for locale "
183                         << QTextCodec::locale() << std::endl;
184         } else
185                 lyxerr[Debug::GUI]
186                         << "Could not find  Qt translations for locale "
187                         << QTextCodec::locale() << std::endl;
188
189 /*#ifdef Q_WS_MACX
190         // These translations are meant to break Qt/Mac menu merging
191         // algorithm on some entries. It lists the menu names that
192         // should not be moved to the LyX menu
193         static QTranslator aqua_trans(0);
194         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Setting", 0,
195                                              "do_not_merge_me"));
196         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Config", 0,
197                                              "do_not_merge_me"));
198         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Options", 0,
199                                              "do_not_merge_me"));
200         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Setup", 0,
201                                              "do_not_merge_me"));
202
203         app.installTranslator(&aqua_trans);
204 #endif
205 */
206         using namespace lyx::graphics;
207
208         Image::newImage = boost::bind(&QLImage::newImage);
209         Image::loadableFormats = boost::bind(&QLImage::loadableFormats);
210
211         // needs to be done before reading lyxrc
212         lyxrc.dpi = getDPI();
213
214         LoaderQueue::setPriority(10,100);
215 }
216
217
218 void parse_lyxrc()
219 {}
220
221
222 void start(string const & batch, vector<string> const & files)
223 {
224         // this can't be done before because it needs the Languages object
225         initEncodings();
226
227         // initial geometry
228         unsigned int width = 690;
229         unsigned int height = 510;
230         // first try lyxrc
231         if (lyxrc.geometry_width != 0 && lyxrc.geometry_height != 0 ) {
232                 width = lyxrc.geometry_width;
233                 height = lyxrc.geometry_height;
234         }
235         // if lyxrc returns (0,0), then use session info
236         else {
237                 string val = LyX::ref().session().loadSessionInfo("WindowWidth");
238                 if (val != "")
239                         width = convert<unsigned int>(val);
240                 val = LyX::ref().session().loadSessionInfo("WindowHeight");
241                 if (val != "")
242                         height = convert<unsigned int>(val);
243         }
244
245         boost::shared_ptr<QtView> view_ptr(new QtView(width, height));
246         LyX::ref().addLyXView(view_ptr);
247
248         QtView & view = *view_ptr.get();
249
250         // if user wants to restore window position
251         if (lyxrc.geometry_xysaved) {
252                 QPoint p = view.pos();
253                 string val = LyX::ref().session().loadSessionInfo("WindowPosX");
254                 if (val != "")
255                         p.setX(convert<unsigned int>(val));
256                 val = LyX::ref().session().loadSessionInfo("WindowPosY");
257                 if (val != "")
258                         p.setY(convert<unsigned int>(val));
259                 view.move(p);
260         }
261         view.show();
262         view.init();
263
264         // FIXME: some code below needs moving
265
266         lyxserver = new LyXServer(&view.getLyXFunc(), lyxrc.lyxpipes);
267         lyxsocket = new LyXServerSocket(&view.getLyXFunc(),
268                           os::internal_path(package().temp_dir() + "/lyxsocket"));
269
270         for_each(files.begin(), files.end(),
271                  bind(&BufferView::loadLyXFile, view.view(), _1, true));
272
273         // handle the batch commands the user asked for
274         if (!batch.empty()) {
275                 view.getLyXFunc().dispatch(lyxaction.lookupFunc(batch));
276         }
277
278         qApp->exec();
279
280         // FIXME
281         delete lyxsocket;
282         delete lyxserver;
283         lyxserver = 0;
284         delete app;
285 }
286
287
288 void sync_events()
289 {
290         // This is the ONLY place where processEvents may be called.
291         // During screen update/ redraw, this method is disabled to
292         // prevent keyboard events being handed to the LyX core, where
293         // they could cause re-entrant calls to screen update.
294 #if QT_VERSION >= 0x030100
295         qApp->processEvents(QEventLoop::ExcludeUserInput);
296 #endif
297 }
298
299
300 void exit()
301 {
302         delete lyxsocket;
303         delete lyxserver;
304         lyxserver = 0;
305
306         // we cannot call qApp->exit(0) - that could return us
307         // into a static dialog return in the lyx code (for example,
308         // load autosave file QMessageBox. We have to just get the hell
309         // out.
310
311         ::exit(0);
312 }
313
314
315 FuncStatus getStatus(FuncRequest const & ev)
316 {
317         FuncStatus flag;
318         switch (ev.action) {
319         case LFUN_DIALOG_SHOW:
320                 if (ev.argument == "preamble")
321                         flag.unknown(true);
322                 break;
323         case LFUN_TOOLTIPS_TOGGLE:
324                 flag.unknown(true);
325                 break;
326         default:
327                 break;
328         }
329
330         return flag;
331 }
332
333
334 bool getRGBColor(LColor_color col, lyx::RGBColor & rgbcol)
335 {
336         QColor const & qcol = lcolorcache.get(col);
337         if (!qcol.isValid()) {
338                 rgbcol.r = 0;
339                 rgbcol.g = 0;
340                 rgbcol.b = 0;
341                 return false;
342         }
343         rgbcol.r = qcol.red();
344         rgbcol.g = qcol.green();
345         rgbcol.b = qcol.blue();
346         return true;
347 }
348
349
350 string const hexname(LColor_color col)
351 {
352         return ltrim(fromqstr(lcolorcache.get(col).name()), "#");
353 }
354
355
356 void update_color(LColor_color)
357 {
358         // FIXME: Bleh, can't we just clear them all at once ?
359         lcolorcache.clear();
360 }
361
362
363 void update_fonts()
364 {
365         fontloader.update();
366 }
367
368
369 bool font_available(LyXFont const & font)
370 {
371         return fontloader.available(font);
372 }
373
374
375 void register_socket_callback(int fd, boost::function<void()> func)
376 {
377         socket_callbacks[fd] = shared_ptr<socket_callback>(new socket_callback(fd, func));
378 }
379
380
381 void unregister_socket_callback(int fd)
382 {
383         socket_callbacks.erase(fd);
384 }
385
386
387 string const roman_font_name()
388 {
389         if (!use_gui)
390                 return "serif";
391
392         QFont font;
393         font.setStyleHint(QFont::Serif);
394         font.setFamily("serif");
395
396         return fromqstr(QFontInfo(font).family());
397 }
398
399
400 string const sans_font_name()
401 {
402         if (!use_gui)
403                 return "sans";
404
405         QFont font;
406         font.setStyleHint(QFont::SansSerif);
407         font.setFamily("sans");
408
409         return fromqstr(QFontInfo(font).family());
410 }
411
412
413 string const typewriter_font_name()
414 {
415         if (!use_gui)
416                 return "monospace";
417
418         QFont font;
419         font.setStyleHint(QFont::TypeWriter);
420         font.setFamily("monospace");
421
422         return fromqstr(QFontInfo(font).family());
423 }
424
425 }; // namespace lyx_gui