*/
#include <config.h>
-#include <gtkmm.h>
-#include "lyx_gui.h"
+// Too hard to make concept checks work with this file
+#ifdef _GLIBCXX_CONCEPT_CHECKS
+#undef _GLIBCXX_CONCEPT_CHECKS
+#endif
+#ifdef _GLIBCPP_CONCEPT_CHECKS
+#undef _GLIBCPP_CONCEPT_CHECKS
+#endif
-#include "support/lyxlib.h"
-#include "support/os.h"
-#include "support/filetools.h"
-#include "support/path_defines.h"
+#include "lyx_gui.h"
#include "debug.h"
#include "funcrequest.h"
#include "gettext.h"
+#include "Color.h"
#include "LColor.h"
#include "LyXAction.h"
#include "lyx_main.h"
#include "lyxfont.h"
#include "graphics/LoaderQueue.h"
+#include "io_callback.h"
+
// FIXME: move this stuff out again
-#include "bufferlist.h"
-#include "buffer_funcs.h"
#include "lyxfunc.h"
#include "lyxserver.h"
#include "lyxsocket.h"
#include "BufferView.h"
+#include "GuiApplication.h"
+#include "GuiImplementation.h"
#include "GView.h"
#include "GtkmmX.h"
-#include "xftFontLoader.h"
#include "GWorkArea.h"
-#include "support/std_sstream.h"
-#include <iomanip>
-#include <fcntl.h>
+#include "support/lyxlib.h"
+#include "support/os.h"
+#include "support/filetools.h"
+#include "support/package.h"
+
+#include <gtkmm.h>
+
#include <boost/bind.hpp>
+#include <boost/function.hpp>
+#include <boost/shared_ptr.hpp>
+
+#include <fcntl.h>
-//just for xforms
-#include "lyx_forms.h"
-#include "xformsImage.h"
-#include "xforms_helpers.h"
+#include <sstream>
+#include <iomanip>
namespace os = lyx::support::os;
using std::ostringstream;
using std::string;
+using lyx::support::package;
-extern BufferList bufferlist;
+using lyx::frontend::colorCache;
+using lyx::frontend::Gui;
+using lyx::frontend::GuiApplication;
+using lyx::frontend::GuiImplementation;
+using lyx::frontend::GView;
-// FIXME: wrong place !
-LyXServer * lyxserver;
-LyXServerSocket * lyxsocket;
bool lyx_gui::use_gui = true;
namespace {
-/// quit lyx
-bool finished = false;
-
-
/// estimate DPI from X server
-float getDPI()
+int getDPI()
{
+ //TODO use GDK instead
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::cref().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;
+ return int(((HeightOfScreen(scr) * 25.4 / HeightMMOfScreen(scr)) +
+ (WidthOfScreen(scr) * 25.4 / WidthMMOfScreen(scr))) / 2);
}
-}
-
-/// read in geometry specification
-char geometry[40];
-
} // namespace anon
+lyx::frontend::Application * theApp;
+GuiApplication * guiApp;
-void parse_init_xforms(int & argc, char * argv[])
+int lyx_gui::exec(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;
+ guiApp = new GuiApplication(argc, argv);
+ theApp = guiApp;
- // connect the image loader based on the xforms library
- Image::newImage = boost::bind(&xformsImage::newImage);
- Image::loadableFormats = boost::bind(&xformsImage::loadableFormats);
+ return LyX::ref().exec2(argc, argv);
}
-void lyx_gui::parse_init(int & argc, char * argv[])
+void lyx_gui::parse_lyxrc()
{
- 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()
+LyXView * lyx_gui::create_view(unsigned int width, unsigned int height,
+ int posx, int posy, bool maximize)
{
- 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);
+ return &guiApp->createView(width, height, posx, posy, maximize);
}
-void lyx_gui::parse_lyxrc()
+int lyx_gui::start(LyXView *, string const & batch)
{
- parse_lyxrc_xforms();
+ return guiApp->start(batch);
}
-void start_xforms()
+void lyx_gui::exit(int status)
{
- // 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;
- }
+ guiApp->exit(status);
}
-void lyx_gui::start(string const & batch, std::vector<string> const & files)
+FuncStatus lyx_gui::getStatus(FuncRequest const & ev)
{
- start_xforms();
- // just for debug
- XSynchronize(getDisplay(), true);
-
- boost::shared_ptr<GView> view_ptr(new GView);
- LyX::ref().addLyXView(view_ptr);
-
- GView & view = *view_ptr.get();
- view.show();
- view.init();
-
- Buffer * last = 0;
-
- // FIXME: some code below needs moving
-
- lyxserver = new LyXServer(&view.getLyXFunc(), lyxrc.lyxpipes);
- lyxsocket = new LyXServerSocket(&view.getLyXFunc(),
- os::slashify_path(os::getTmpDir() + "/lyxsocket"));
-
- std::vector<string>::const_iterator cit = files.begin();
- std::vector<string>::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(lyxaction.lookupFunc(batch));
- }
-
- // enter the event loop
- while (!finished) {
- while (Gtk::Main::events_pending())
- Gtk::Main::iteration(false);
- events_xforms();
+ FuncStatus flag;
+ switch (ev.action) {
+ case LFUN_TOOLTIPS_TOGGLE:
+ flag.unknown(true);
+ break;
+ default:
+ break;
}
- // FIXME: breaks emergencyCleanup
- delete lyxsocket;
- delete lyxserver;
+ return flag;
}
-void lyx_gui::exit()
+bool lyx_gui::getRGBColor(LColor_color col, lyx::RGBColor & rgbcol)
{
- finished = true;
-}
-
+ Gdk::Color gdkColor;
+ Gdk::Color * gclr = colorCache.getColor(col);
+ if (!gclr) {
+ gclr = &gdkColor;
+ if(!gclr->parse(lcolor.getX11Name(col))) {
+ rgbcol.r = 0;
+ rgbcol.g = 0;
+ rgbcol.b = 0;
+ return false;
+ }
+ }
-FuncStatus lyx_gui::getStatus(FuncRequest const & /*ev*/)
-{
- // Nothing interesting to do here
- return FuncStatus();
+ // Note that X stores the RGB values in the range 0 - 65535
+ // whilst we require them in the range 0 - 255.
+ rgbcol.r = gclr->get_red() / 256;
+ rgbcol.g = gclr->get_green() / 256;
+ rgbcol.b = gclr->get_blue() / 256;
+ return true;
}
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));
+ lyx::RGBColor rgbcol;
+ if (!getRGBColor(col, rgbcol)) {
+ lyxerr << "X can't find color for \"" << lcolor.getLyXName(col)
+ << '"' << std::endl;
+ return string();
}
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);
+ << std::setw(2) << rgbcol.r
+ << std::setw(2) << rgbcol.g
+ << std::setw(2) << rgbcol.b;
return os.str();
}
namespace {
+std::map<int, boost::shared_ptr<io_callback> > callbacks;
-bool readCallback(Glib::IOCondition /*condition*/, LyXComm * comm)
-{
- comm->read_ready();
- return true;
-}
-
+} // NS anon
-std::map<int, SigC::Connection> gReadCallbackMap;
-
-}
-
-void lyx_gui::set_read_callback(int fd, LyXComm * comm)
+void lyx_gui::register_socket_callback(int fd,
+ boost::function<void()> func)
{
- gReadCallbackMap[fd] = Glib::signal_io().connect(
- SigC::bind(SigC::slot(readCallback), comm),
- fd,
- Glib::IO_IN);
+ callbacks[fd] = boost::shared_ptr<io_callback>(new io_callback(fd, func));
}
-void lyx_gui::remove_read_callback(int fd)
+void lyx_gui::unregister_socket_callback(int fd)
{
- gReadCallbackMap[fd].disconnect();
- gReadCallbackMap.erase(fd);
+ callbacks.erase(fd);
}
-void lyx_gui::set_datasocket_callback(LyXDataSocket * /* p */)
-{}
-
-
-void lyx_gui::remove_datasocket_callback(LyXDataSocket * /* p */)
-{}
-
-
-void lyx_gui::set_serversocket_callback(LyXServerSocket * /* p */)
-{}
-
-
-void lyx_gui::remove_serversocket_callback(LyXServerSocket * /* p */)
-{}
-
-
string const lyx_gui::roman_font_name()
{
return "times";