2003-09-02 10:29:05 +00:00
|
|
|
/**
|
|
|
|
* \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
|
|
|
|
*
|
2003-09-02 17:02:32 +00:00
|
|
|
* Full author contact details are available in file CREDITS.
|
2003-09-02 10:29:05 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <gtkmm.h>
|
|
|
|
|
|
|
|
#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"
|
|
|
|
|
2003-09-05 17:23:11 +00:00
|
|
|
#include "support/std_sstream.h"
|
2003-09-02 10:29:05 +00:00
|
|
|
#include <iomanip>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <boost/bind.hpp>
|
|
|
|
|
|
|
|
//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<string> 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<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(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<int, SigC::Connection> 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
|
|
|
|
}
|