/** * \file gtk/lyx_gui.C * This file is part of LyX, the document processor. * Licence details can be found in the file COPYING. * * \author Lars Gullik Bjnes * \author John Levon * \author Huang Ying * * Full author contact details are available in file CREDITS. */ #include #include #include "lyx_gui.h" #include "support/lyxlib.h" #include "support/os.h" #include "support/filetools.h" #include "support/path_defines.h" #include "debug.h" #include "gettext.h" #include "lyx_main.h" #include "lyxrc.h" #include "lyxfont.h" #include "graphics/LoaderQueue.h" // FIXME: move this stuff out again #include "bufferlist.h" #include "buffer_funcs.h" #include "lyxfunc.h" #include "lyxserver.h" #include "BufferView.h" #include "GView.h" #include "GtkmmX.h" #include "xftFontLoader.h" #include "GWorkArea.h" #include "support/std_sstream.h" #include #include #include //just for xforms #include "lyx_forms.h" #include "xformsImage.h" #include "xforms_helpers.h" extern BufferList bufferlist; // FIXME: wrong place ! LyXServer * lyxserver; bool lyx_gui::use_gui = true; namespace { /// quit lyx bool finished = false; /// estimate DPI from X server float getDPI() { Screen * scr = ScreenOfDisplay(getDisplay(), getScreen()); return ((HeightOfScreen(scr) * 25.4 / HeightMMOfScreen(scr)) + (WidthOfScreen(scr) * 25.4 / WidthMMOfScreen(scr))) / 2; } /// set default GUI configuration void setDefaults() { FL_IOPT cntl; cntl.buttonFontSize = FL_NORMAL_SIZE; cntl.browserFontSize = FL_NORMAL_SIZE; cntl.labelFontSize = FL_NORMAL_SIZE; cntl.choiceFontSize = FL_NORMAL_SIZE; cntl.inputFontSize = FL_NORMAL_SIZE; cntl.menuFontSize = FL_NORMAL_SIZE; cntl.borderWidth = -1; cntl.vclass = FL_DefaultVisual; fl_set_defaults(FL_PDVisual | FL_PDButtonFontSize | FL_PDBrowserFontSize | FL_PDLabelFontSize | FL_PDChoiceFontSize | FL_PDInputFontSize | FL_PDMenuFontSize | FL_PDBorderWidth, &cntl); } extern "C" { int LyX_XErrHandler(Display * display, XErrorEvent * xeev) { // We don't abort on BadWindow if (xeev->error_code == BadWindow) { lyxerr << "BadWindow received !" << std::endl; lyxerr << "If you're using xforms 1.0 or greater, " << " please report this to lyx-devel@lists.lyx.org" << std::endl; return 0; } // emergency cleanup LyX::emergencyCleanup(); // Get the reason for the crash. char etxt[513]; XGetErrorText(display, xeev->error_code, etxt, 512); lyxerr << etxt << " id: " << xeev->resourceid << std::endl; // By doing an abort we get a nice backtrace. (hopefully) lyx::support::abort(); return 0; } } /// read in geometry specification char geometry[40]; } // namespace anon void parse_init_xforms(int & argc, char * argv[]) { setDefaults(); FL_CMD_OPT cmdopt[] = { {"-geometry", "*.geometry", XrmoptionSepArg, "690x510"} }; FL_resource res[] = { {"geometry", "geometryClass", FL_STRING, geometry, "", 40} }; const int num_res = sizeof(res)/sizeof(FL_resource); fl_initialize(&argc, argv, "LyX", cmdopt, num_res); // It appears that, in xforms >=0.89.5, fl_initialize() // calls setlocale() and ruins our LC_NUMERIC setting. fl_get_app_resources(res, num_res); Display * display = fl_get_display(); if (!display) { lyxerr << "LyX: unable to access X display, exiting" << std::endl; lyx::support::os::warn("Unable to access X display, exiting"); ::exit(1); } fcntl(ConnectionNumber(display), F_SETFD, FD_CLOEXEC); XSetErrorHandler(LyX_XErrHandler); using namespace lyx::graphics; // connect the image loader based on the xforms library Image::newImage = boost::bind(&xformsImage::newImage); Image::loadableFormats = boost::bind(&xformsImage::loadableFormats); } void lyx_gui::parse_init(int & argc, char * argv[]) { new Gtk::Main(argc, argv); parse_init_xforms(argc, argv); locale_init(); // must do this /before/ lyxrc gets read lyxrc.dpi = getDPI(); } void parse_lyxrc_xforms() { XformsColor::read(lyx::support::AddName( lyx::support::user_lyxdir(), "preferences.xform")); if (lyxrc.popup_font_encoding.empty()) lyxrc.popup_font_encoding = lyxrc.font_norm; // Set the font name for popups and menus string boldfontname = lyxrc.popup_bold_font + "-*-*-*-?-*-*-*-*-" + lyxrc.popup_font_encoding; // "?" means "scale that font" string fontname = lyxrc.popup_normal_font + "-*-*-*-?-*-*-*-*-" + lyxrc.popup_font_encoding; int bold = fl_set_font_name(FL_BOLD_STYLE, boldfontname.c_str()); int normal = fl_set_font_name(FL_NORMAL_STYLE, fontname.c_str()); if (bold < 0) lyxerr << "Could not set menu font to " << boldfontname << std::endl; if (normal < 0) lyxerr << "Could not set popup font to " << fontname << std::endl; if (bold < 0 && normal < 0) { lyxerr << "Using 'helvetica' font for menus" << std::endl; boldfontname = "-*-helvetica-bold-r-*-*-*-?-*-*-*-*-iso8859-1"; fontname = "-*-helvetica-medium-r-*-*-*-?-*-*-*-*-iso8859-1"; bold = fl_set_font_name(FL_BOLD_STYLE, boldfontname.c_str()); normal = fl_set_font_name(FL_NORMAL_STYLE, fontname.c_str()); if (bold < 0 && normal < 0) { lyxerr << "Could not find helvetica font. Using 'fixed'." << std::endl; fl_set_font_name(FL_NORMAL_STYLE, "fixed"); normal = bold = 0; } } if (bold < 0) fl_set_font_name(FL_BOLD_STYLE, fontname.c_str()); else if (normal < 0) fl_set_font_name(FL_NORMAL_STYLE, boldfontname.c_str()); fl_setpup_fontstyle(FL_NORMAL_STYLE); fl_setpup_fontsize(FL_NORMAL_SIZE); fl_setpup_color(FL_MCOL, FL_BLACK); fl_set_goodies_font(FL_NORMAL_STYLE, FL_NORMAL_SIZE); fl_set_tooltip_font(FL_NORMAL_STYLE, FL_NORMAL_SIZE); } void lyx_gui::parse_lyxrc() { parse_lyxrc_xforms(); } void start_xforms() { // initial geometry int xpos = -1; int ypos = -1; unsigned int width = 690; unsigned int height = 510; int const geometryBitmask = XParseGeometry(geometry, &xpos, &ypos, &width, &height); // if width is not set by geometry, check it against monitor width if (!(geometryBitmask & WidthValue)) { Screen * scr = ScreenOfDisplay(fl_get_display(), fl_screen); if (WidthOfScreen(scr) - 8 < int(width)) width = WidthOfScreen(scr) - 8; } // if height is not set by geometry, check it against monitor height if (!(geometryBitmask & HeightValue)) { Screen * scr = ScreenOfDisplay(fl_get_display(), fl_screen); if (HeightOfScreen(scr) - 24 < int(height)) height = HeightOfScreen(scr) - 24; } Screen * s = ScreenOfDisplay(fl_get_display(), fl_screen); // recalculate xpos if it's not set if (xpos == -1) xpos = (WidthOfScreen(s) - width) / 2; // recalculate ypos if it's not set if (ypos == -1) ypos = (HeightOfScreen(s) - height) / 2; lyxerr[Debug::GUI] << "Creating view: " << width << 'x' << height << '+' << xpos << '+' << ypos << std::endl; // XFormsView view(width, height); // view.show(xpos, ypos, "LyX"); // view.init(); } static void events_xforms() { if (fl_check_forms() == FL_EVENT) { XEvent ev; fl_XNextEvent(&ev); lyxerr[Debug::GUI] << "Received unhandled X11 event" << std::endl << "Type: " << ev.xany.type << " Target: 0x" << std::hex << ev.xany.window << std::dec << std::endl; } } void lyx_gui::start(string const & batch, std::vector const & files) { start_xforms(); // just for debug XSynchronize(getDisplay(), true); GView view; view.show(); view.init(); Buffer * last = 0; // FIXME: some code below needs moving lyxserver = new LyXServer(&view.getLyXFunc(), lyxrc.lyxpipes); std::vector::const_iterator cit = files.begin(); std::vector::const_iterator end = files.end(); for (; cit != end; ++cit) { Buffer * b = bufferlist.newBuffer(*cit); if (loadLyXFile(b, *cit)) last = b; } // switch to the last buffer successfully loaded if (last) { view.view()->buffer(last); } // handle the batch commands the user asked for if (!batch.empty()) { view.getLyXFunc().dispatch(batch); } // enter the event loop while (!finished) { while (Gtk::Main::events_pending()) Gtk::Main::iteration(false); events_xforms(); } // FIXME: breaks emergencyCleanup delete lyxserver; } void lyx_gui::exit() { finished = true; } FuncStatus lyx_gui::getStatus(FuncRequest const & /*ev*/) { // Nothing interesting to do here return FuncStatus(); } string const lyx_gui::hexname(LColor::color col) { Gdk::Color gdkColor; Gdk::Color * gclr = colorCache.getColor(col); if (!gclr) { gclr = &gdkColor; gclr->parse(lcolor.getX11Name(col)); } std::ostringstream os; // Note that X stores the RGB values in the range 0 - 65535 // whilst we require them in the range 0 - 255. os << std::setbase(16) << std::setfill('0') << std::setw(2) << (gclr->get_red() / 256) << std::setw(2) << (gclr->get_green() / 256) << std::setw(2) << (gclr->get_blue() / 256); return os.str(); } void lyx_gui::update_color(LColor::color /*col*/) { colorCache.clear(); } void lyx_gui::update_fonts() { fontLoader.update(); } bool lyx_gui::font_available(LyXFont const & font) { return fontLoader.available(font); } namespace { bool readCallback(Glib::IOCondition /*condition*/, LyXComm * comm) { comm->read_ready(); return true; } std::map gReadCallbackMap; } void lyx_gui::set_read_callback(int fd, LyXComm * comm) { gReadCallbackMap[fd] = Glib::signal_io().connect( SigC::bind(SigC::slot(readCallback), comm), fd, Glib::IO_IN); } void lyx_gui::remove_read_callback(int fd) { gReadCallbackMap[fd].disconnect(); gReadCallbackMap.erase(fd); } string const lyx_gui::roman_font_name() { return "times"; } string const lyx_gui::sans_font_name() { return "helvetica"; } string const lyx_gui::typewriter_font_name() { return "courier"; } void lyx_gui::sync_events() { // FIXME }