]> git.lyx.org Git - lyx.git/blob - src/frontends/xforms/lyx_gui.C
Alfredo's setPriority patch to the graphics LoaderQueue.
[lyx.git] / src / frontends / xforms / lyx_gui.C
1 /**
2  * \file xforms/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 #include "support/lyxlib.h"
17 #include "support/os.h"
18 #include "support/filetools.h"
19
20 #include "debug.h"
21 #include "gettext.h"
22
23 #include "lyx_main.h"
24 #include "lyxrc.h"
25 #include "lyxfont.h"
26 #include "graphics/LoaderQueue.h"
27
28 // FIXME: move this stuff out again
29 #include "bufferlist.h"
30 #include "lyxfunc.h"
31 #include "lyxserver.h"
32 #include "BufferView.h"
33 #include "XFormsView.h"
34
35 #include FORMS_H_LOCATION
36 #include "ColorHandler.h"
37 #include "xforms_helpers.h"
38 #include "xfont_loader.h"
39 #include "xformsImage.h"
40
41 #include "Lsstream.h"
42 #include <iomanip>
43 #include <fcntl.h>
44 #include <boost/bind.hpp>
45
46 #ifndef CXX_GLOBAL_CSTD
47 using std::exit;
48 #endif
49
50 using std::vector;
51 using std::hex;
52 using std::dec;
53 using std::endl;
54 using std::setbase;
55 using std::setfill;
56 using std::setw;
57
58 extern BufferList bufferlist;
59
60 // FIXME: wrong place !
61 LyXServer * lyxserver;
62
63 namespace {
64
65 /// quit lyx
66 bool finished = false;
67
68 /// estimate DPI from X server
69 float getDPI()
70 {
71         Screen * scr = ScreenOfDisplay(fl_get_display(), fl_screen);
72         return ((HeightOfScreen(scr) * 25.4 / HeightMMOfScreen(scr)) +
73                 (WidthOfScreen(scr) * 25.4 / WidthMMOfScreen(scr))) / 2;
74 }
75
76
77 /// set default GUI configuration
78 void setDefaults()
79 {
80         FL_IOPT cntl;
81         cntl.buttonFontSize = FL_NORMAL_SIZE;
82         cntl.browserFontSize = FL_NORMAL_SIZE;
83         cntl.labelFontSize = FL_NORMAL_SIZE;
84         cntl.choiceFontSize = FL_NORMAL_SIZE;
85         cntl.inputFontSize = FL_NORMAL_SIZE;
86         cntl.menuFontSize  = FL_NORMAL_SIZE;
87         cntl.borderWidth = -1;
88         cntl.vclass = FL_DefaultVisual;
89         fl_set_defaults(FL_PDVisual
90                         | FL_PDButtonFontSize
91                         | FL_PDBrowserFontSize
92                         | FL_PDLabelFontSize
93                         | FL_PDChoiceFontSize
94                         | FL_PDInputFontSize
95                         | FL_PDMenuFontSize
96                         | FL_PDBorderWidth, &cntl);
97 }
98
99
100 extern "C" {
101
102 int LyX_XErrHandler(Display * display, XErrorEvent * xeev) {
103         // We don't abort on BadWindow
104         if (xeev->error_code == BadWindow) {
105                 lyxerr << "BadWindow received !" << endl;
106                 lyxerr << "If you're using xforms 1.0 or greater, "
107                         << " please report this to lyx-devel@lists.lyx.org" << endl;
108                 return 0;
109         }
110
111         // emergency cleanup
112         LyX::emergencyCleanup();
113
114         // Get the reason for the crash.
115         char etxt[513];
116         XGetErrorText(display, xeev->error_code, etxt, 512);
117         lyxerr << etxt << " id: " << xeev->resourceid << endl;
118         // By doing an abort we get a nice backtrace. (hopefully)
119         lyx::abort();
120         return 0;
121 }
122
123 }
124
125 /// read in geometry specification
126 char geometry[40];
127
128 } // namespace anon
129
130
131 void lyx_gui::parse_init(int & argc, char * argv[])
132 {
133         setDefaults();
134
135         FL_CMD_OPT cmdopt[] = {
136                 {"-geometry", "*.geometry", XrmoptionSepArg, "690x510"}
137         };
138
139         FL_resource res[] = {
140                 {"geometry", "geometryClass", FL_STRING, geometry, "", 40}
141         };
142
143         const int num_res = sizeof(res)/sizeof(FL_resource);
144
145         fl_initialize(&argc, argv, "LyX", cmdopt, num_res);
146
147         // It appears that, in xforms >=0.89.5, fl_initialize()
148         // calls setlocale() and ruins our LC_NUMERIC setting.
149         locale_init();
150
151         fl_get_app_resources(res, num_res);
152
153         Display * display = fl_get_display();
154
155         if (!display) {
156                 lyxerr << "LyX: unable to access X display, exiting" << endl;
157                 os::warn("Unable to access X display, exiting");
158                 ::exit(1);
159         }
160
161         fcntl(ConnectionNumber(display), F_SETFD, FD_CLOEXEC);
162
163         XSetErrorHandler(LyX_XErrHandler);
164
165         lyxColorHandler.reset(new LyXColorHandler());
166
167         using namespace grfx;
168
169         // connect the image loader based on the xforms library
170         Image::newImage = boost::bind(&xformsImage::newImage);
171         Image::loadableFormats = boost::bind(&xformsImage::loadableFormats);
172
173         // must do this /before/ lyxrc gets read
174         lyxrc.dpi = getDPI();
175
176         LoaderQueue::setPriority(10,100);
177 }
178
179
180 void lyx_gui::parse_lyxrc()
181 {
182         XformsColor::read(AddName(user_lyxdir, "preferences.xform"));
183
184         if (lyxrc.popup_font_encoding.empty())
185                 lyxrc.popup_font_encoding = lyxrc.font_norm;
186         // Set the font name for popups and menus
187         string boldfontname = lyxrc.popup_bold_font
188                                + "-*-*-*-?-*-*-*-*-"
189                                + lyxrc.popup_font_encoding;
190                 // "?" means "scale that font"
191         string fontname = lyxrc.popup_normal_font
192                                + "-*-*-*-?-*-*-*-*-"
193                                + lyxrc.popup_font_encoding;
194
195         int bold = fl_set_font_name(FL_BOLD_STYLE, boldfontname.c_str());
196         int normal = fl_set_font_name(FL_NORMAL_STYLE, fontname.c_str());
197         if (bold < 0)
198                 lyxerr << "Could not set menu font to "
199                        << boldfontname << endl;
200
201         if (normal < 0)
202                 lyxerr << "Could not set popup font to "
203                        << fontname << endl;
204
205         if (bold < 0 && normal < 0) {
206                 lyxerr << "Using 'helvetica' font for menus" << endl;
207                 boldfontname = "-*-helvetica-bold-r-*-*-*-?-*-*-*-*-iso8859-1";
208                 fontname = "-*-helvetica-medium-r-*-*-*-?-*-*-*-*-iso8859-1";
209                 bold = fl_set_font_name(FL_BOLD_STYLE, boldfontname.c_str());
210                 normal = fl_set_font_name(FL_NORMAL_STYLE, fontname.c_str());
211
212                 if (bold < 0 && normal < 0) {
213                         lyxerr << "Could not find helvetica font. Using 'fixed'." << endl;
214                         fl_set_font_name(FL_NORMAL_STYLE, "fixed");
215                         normal = bold = 0;
216                 }
217         }
218         if (bold < 0)
219                 fl_set_font_name(FL_BOLD_STYLE, fontname.c_str());
220         else if (normal < 0)
221                 fl_set_font_name(FL_NORMAL_STYLE, boldfontname.c_str());
222
223         fl_setpup_fontstyle(FL_NORMAL_STYLE);
224         fl_setpup_fontsize(FL_NORMAL_SIZE);
225         fl_setpup_color(FL_MCOL, FL_BLACK);
226         fl_set_goodies_font(FL_NORMAL_STYLE, FL_NORMAL_SIZE);
227         fl_set_tooltip_font(FL_NORMAL_STYLE, FL_NORMAL_SIZE);
228 }
229
230
231 void lyx_gui::start(string const & batch, vector<string> const & files)
232 {
233         // initial geometry
234         int xpos = -1;
235         int ypos = -1;
236         unsigned int width = 690;
237         unsigned int height = 510;
238
239         int const geometryBitmask =
240                 XParseGeometry(geometry,
241                                &xpos, &ypos, &width, &height);
242
243         // if width is not set by geometry, check it against monitor width
244         if (!(geometryBitmask & WidthValue)) {
245                 Screen * scr = ScreenOfDisplay(fl_get_display(), fl_screen);
246                 if (WidthOfScreen(scr) - 8 < int(width))
247                         width = WidthOfScreen(scr) - 8;
248         }
249
250         // if height is not set by geometry, check it against monitor height
251         if (!(geometryBitmask & HeightValue)) {
252                 Screen * scr = ScreenOfDisplay(fl_get_display(), fl_screen);
253                 if (HeightOfScreen(scr) - 24 < int(height))
254                         height = HeightOfScreen(scr) - 24;
255         }
256
257         Screen * s = ScreenOfDisplay(fl_get_display(), fl_screen);
258
259         // recalculate xpos if it's not set
260         if (xpos == -1)
261                 xpos = (WidthOfScreen(s) - width) / 2;
262
263         // recalculate ypos if it's not set
264         if (ypos == -1)
265                 ypos = (HeightOfScreen(s) - height) / 2;
266
267         lyxerr[Debug::GUI] << "Creating view: " << width << 'x' << height
268                            << '+' << xpos << '+' << ypos << endl;
269
270         XFormsView view(width, height);
271         view.show(xpos, ypos, "LyX");
272         view.init();
273
274         Buffer * last = 0;
275
276         // FIXME: some code below needs moving
277
278         lyxserver = new LyXServer(&view.getLyXFunc(), lyxrc.lyxpipes);
279
280         vector<string>::const_iterator cit = files.begin();
281         vector<string>::const_iterator end = files.end();
282         for (; cit != end; ++cit) {
283                 Buffer * b = bufferlist.loadLyXFile(*cit);
284                 if (b) {
285                         last = b;
286                 }
287         }
288
289         // switch to the last buffer successfully loaded
290         if (last) {
291                 view.view()->buffer(last);
292         }
293
294         // handle the batch commands the user asked for
295         if (!batch.empty()) {
296                 view.getLyXFunc().dispatch(batch);
297         }
298
299         // enter the event loop
300         while (!finished) {
301                 if (fl_check_forms() == FL_EVENT) {
302                         XEvent ev;
303                         fl_XNextEvent(&ev);
304                         lyxerr[Debug::GUI]
305                                 << "Received unhandled X11 event" << endl
306                                 << "Type: " << ev.xany.type
307                                 << " Target: 0x" << hex << ev.xany.window
308                                 << dec << endl;
309                 }
310         }
311
312         // FIXME: breaks emergencyCleanup
313         delete lyxserver;
314 }
315
316
317 void lyx_gui::exit()
318 {
319         finished = true;
320 }
321
322
323 FuncStatus lyx_gui::getStatus(FuncRequest const & /*ev*/)
324 {
325         // Nothing interesting to do here
326         return FuncStatus();
327 }
328
329 string const lyx_gui::hexname(LColor::color col)
330 {
331         unsigned int r, g, b;
332         bool const success = getRGBColor(col, r, g, b);
333         if (!success) {
334                 lyxerr << "X can't find color for \"" << lcolor.getLyXName(col)
335                        << '"' << endl;
336                 return string();
337         }
338
339         ostringstream os;
340
341         os << setbase(16) << setfill('0')
342            << setw(2) << r
343            << setw(2) << g
344            << setw(2) << b;
345
346         return STRCONV(os.str());
347 }
348
349
350 void lyx_gui::update_color(LColor::color col)
351 {
352         lyxColorHandler->updateColor(col);
353 }
354
355
356 void lyx_gui::update_fonts()
357 {
358         fontloader.update();
359 }
360
361
362 bool lyx_gui::font_available(LyXFont const & font)
363 {
364         return fontloader.available(font);
365 }
366
367 namespace {
368
369 extern "C"
370 void C_read_callback(int, void * data)
371 {
372         LyXComm * comm = static_cast<LyXComm *>(data);
373         comm->read_ready();
374 }
375
376 }
377
378 void lyx_gui::set_read_callback(int fd, LyXComm * comm)
379 {
380         fl_add_io_callback(fd, FL_READ, C_read_callback, comm);
381 }
382
383
384 void lyx_gui::remove_read_callback(int fd)
385 {
386         fl_remove_io_callback(fd, FL_READ, C_read_callback);
387 }