]> git.lyx.org Git - lyx.git/blob - src/frontends/qt4/lyx_gui.C
fix LFUN enum values (some of them were broken by r13801)
[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 lyx_gui {
156
157 bool use_gui = true;
158
159 void parse_init(int & argc, char * argv[])
160 {
161         // Force adding of font path _before_ QApplication is initialized
162         FontLoader::initFontPath();
163
164         static LQApplication app(argc, argv);
165
166         // install translation file for Qt built-in dialogs
167         // These are only installed since Qt 3.2.x
168         static QTranslator qt_trans(0);
169         if (qt_trans.load(QString("qt_") + QTextCodec::locale(),
170                           qInstallPathTranslations())) {
171                 app.installTranslator(&qt_trans);
172                 // even if the language calls for RtL, don't do that
173                 app.setReverseLayout(false);
174                 lyxerr[Debug::GUI]
175                         << "Successfully installed Qt translations for locale "
176                         << QTextCodec::locale() << std::endl;
177         } else
178                 lyxerr[Debug::GUI]
179                         << "Could not find  Qt translations for locale "
180                         << QTextCodec::locale() << std::endl;
181
182 /*#ifdef Q_WS_MACX
183         // These translations are meant to break Qt/Mac menu merging
184         // algorithm on some entries. It lists the menu names that
185         // should not be moved to the LyX menu
186         static QTranslator aqua_trans(0);
187         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Setting", 0,
188                                              "do_not_merge_me"));
189         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Config", 0,
190                                              "do_not_merge_me"));
191         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Options", 0,
192                                              "do_not_merge_me"));
193         aqua_trans.insert(QTranslatorMessage("QMenuBar", "Setup", 0,
194                                              "do_not_merge_me"));
195
196         app.installTranslator(&aqua_trans);
197 #endif
198 */
199         using namespace lyx::graphics;
200
201         Image::newImage = boost::bind(&QLImage::newImage);
202         Image::loadableFormats = boost::bind(&QLImage::loadableFormats);
203
204         // needs to be done before reading lyxrc
205         lyxrc.dpi = getDPI();
206
207         LoaderQueue::setPriority(10,100);
208 }
209
210
211 void parse_lyxrc()
212 {}
213
214
215 void start(string const & batch, vector<string> const & files)
216 {
217         // this can't be done before because it needs the Languages object
218         initEncodings();
219
220         // initial geometry
221         unsigned int width = 690;
222         unsigned int height = 510;
223         // first try lyxrc
224         if (lyxrc.geometry_width != 0 && lyxrc.geometry_height != 0 ) {
225                 width = lyxrc.geometry_width;
226                 height = lyxrc.geometry_height;
227         }
228         // if lyxrc returns (0,0), then use session info
229         else {
230                 string val = LyX::ref().session().loadSessionInfo("WindowWidth");
231                 if (val != "")
232                         width = convert<unsigned int>(val);
233                 val = LyX::ref().session().loadSessionInfo("WindowHeight");
234                 if (val != "")
235                         height = convert<unsigned int>(val);
236         }
237
238         boost::shared_ptr<QtView> view_ptr(new QtView(width, height));
239         LyX::ref().addLyXView(view_ptr);
240
241         QtView & view = *view_ptr.get();
242
243         // if user wants to restore window position
244         if (lyxrc.geometry_xysaved) {
245                 QPoint p = view.pos();
246                 string val = LyX::ref().session().loadSessionInfo("WindowPosX");
247                 if (val != "")
248                         p.setX(convert<unsigned int>(val));
249                 val = LyX::ref().session().loadSessionInfo("WindowPosY");
250                 if (val != "")
251                         p.setY(convert<unsigned int>(val));
252                 view.move(p);
253         }
254         view.show();
255         view.init();
256
257         // FIXME: some code below needs moving
258
259         lyxserver = new LyXServer(&view.getLyXFunc(), lyxrc.lyxpipes);
260         lyxsocket = new LyXServerSocket(&view.getLyXFunc(),
261                           os::internal_path(package().temp_dir() + "/lyxsocket"));
262
263         for_each(files.begin(), files.end(),
264                  bind(&BufferView::loadLyXFile, view.view(), _1, true));
265
266         // handle the batch commands the user asked for
267         if (!batch.empty()) {
268                 view.getLyXFunc().dispatch(lyxaction.lookupFunc(batch));
269         }
270
271         qApp->exec();
272
273         // FIXME
274         delete lyxsocket;
275         delete lyxserver;
276         lyxserver = 0;
277 }
278
279
280 void sync_events()
281 {
282         // This is the ONLY place where processEvents may be called.
283         // During screen update/ redraw, this method is disabled to
284         // prevent keyboard events being handed to the LyX core, where
285         // they could cause re-entrant calls to screen update.
286 #if QT_VERSION >= 0x030100
287         qApp->processEvents(QEventLoop::ExcludeUserInput);
288 #endif
289 }
290
291
292 void exit()
293 {
294         delete lyxsocket;
295         delete lyxserver;
296         lyxserver = 0;
297
298         // we cannot call qApp->exit(0) - that could return us
299         // into a static dialog return in the lyx code (for example,
300         // load autosave file QMessageBox. We have to just get the hell
301         // out.
302
303         ::exit(0);
304 }
305
306
307 FuncStatus getStatus(FuncRequest const & ev)
308 {
309         FuncStatus flag;
310         switch (ev.action) {
311         case LFUN_DIALOG_SHOW:
312                 if (ev.argument == "preamble")
313                         flag.unknown(true);
314                 break;
315         case LFUN_TOOLTIPS_TOGGLE:
316                 flag.unknown(true);
317                 break;
318         default:
319                 break;
320         }
321
322         return flag;
323 }
324
325
326 bool getRGBColor(LColor_color col, lyx::RGBColor & rgbcol)
327 {
328         QColor const & qcol = lcolorcache.get(col);
329         if (!qcol.isValid()) {
330                 rgbcol.r = 0;
331                 rgbcol.g = 0;
332                 rgbcol.b = 0;
333                 return false;
334         }
335         rgbcol.r = qcol.red();
336         rgbcol.g = qcol.green();
337         rgbcol.b = qcol.blue();
338         return true;
339 }
340
341
342 string const hexname(LColor_color col)
343 {
344         return ltrim(fromqstr(lcolorcache.get(col).name()), "#");
345 }
346
347
348 void update_color(LColor_color)
349 {
350         // FIXME: Bleh, can't we just clear them all at once ?
351         lcolorcache.clear();
352 }
353
354
355 void update_fonts()
356 {
357         fontloader.update();
358 }
359
360
361 bool font_available(LyXFont const & font)
362 {
363         return fontloader.available(font);
364 }
365
366
367 void register_socket_callback(int fd, boost::function<void()> func)
368 {
369         socket_callbacks[fd] = shared_ptr<socket_callback>(new socket_callback(fd, func));
370 }
371
372
373 void unregister_socket_callback(int fd)
374 {
375         socket_callbacks.erase(fd);
376 }
377
378
379 string const roman_font_name()
380 {
381         if (!use_gui)
382                 return "serif";
383
384         QFont font;
385         font.setStyleHint(QFont::Serif);
386         font.setFamily("serif");
387
388         return fromqstr(QFontInfo(font).family());
389 }
390
391
392 string const sans_font_name()
393 {
394         if (!use_gui)
395                 return "sans";
396
397         QFont font;
398         font.setStyleHint(QFont::SansSerif);
399         font.setFamily("sans");
400
401         return fromqstr(QFontInfo(font).family());
402 }
403
404
405 string const typewriter_font_name()
406 {
407         if (!use_gui)
408                 return "monospace";
409
410         QFont font;
411         font.setStyleHint(QFont::TypeWriter);
412         font.setFamily("monospace");
413
414         return fromqstr(QFontInfo(font).family());
415 }
416
417 }; // namespace lyx_gui