2002-06-18 15:44:30 +00:00
|
|
|
/**
|
2006-06-20 08:39:16 +00:00
|
|
|
* \file WorkArea.C
|
2002-09-05 15:14:23 +00:00
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
2002-06-18 15:44:30 +00:00
|
|
|
*
|
2002-11-27 10:30:28 +00:00
|
|
|
* \author John Levon
|
2006-06-20 08:39:16 +00:00
|
|
|
* \author Abdelrazak Younes
|
2002-09-05 14:10:50 +00:00
|
|
|
*
|
2003-08-23 00:17:00 +00:00
|
|
|
* Full author contact details are available in file CREDITS.
|
2002-06-25 15:59:10 +00:00
|
|
|
*
|
|
|
|
* Splash screen code added by Angus Leeming
|
2002-06-18 15:44:30 +00:00
|
|
|
*/
|
1999-09-27 18:44:28 +00:00
|
|
|
|
2002-06-18 15:44:30 +00:00
|
|
|
#include <config.h>
|
2000-02-18 22:22:42 +00:00
|
|
|
|
2006-10-07 16:15:06 +00:00
|
|
|
#include "frontends/WorkArea.h"
|
|
|
|
|
|
|
|
#include "frontends/Application.h"
|
|
|
|
#include "frontends/FontMetrics.h"
|
2006-06-20 08:39:16 +00:00
|
|
|
|
2006-08-15 21:57:23 +00:00
|
|
|
#include "funcrequest.h"
|
|
|
|
#include "lyxfunc.h"
|
2003-09-05 10:55:42 +00:00
|
|
|
#include "Painter.h"
|
2003-07-28 18:40:05 +00:00
|
|
|
|
2001-06-25 00:06:33 +00:00
|
|
|
#include "BufferView.h"
|
2002-06-18 15:44:30 +00:00
|
|
|
#include "buffer.h"
|
2003-09-09 11:24:33 +00:00
|
|
|
#include "bufferparams.h"
|
2005-02-04 07:16:03 +00:00
|
|
|
#include "coordcache.h"
|
2003-11-10 09:06:48 +00:00
|
|
|
#include "cursor.h"
|
2002-06-18 15:44:30 +00:00
|
|
|
#include "debug.h"
|
2003-09-05 10:55:42 +00:00
|
|
|
#include "language.h"
|
2003-09-16 09:44:34 +00:00
|
|
|
#include "LColor.h"
|
2002-06-25 15:59:10 +00:00
|
|
|
#include "lyxfont.h"
|
2003-09-05 10:55:42 +00:00
|
|
|
#include "lyxrc.h"
|
|
|
|
#include "lyxrow.h"
|
|
|
|
#include "lyxtext.h"
|
2006-08-15 21:57:23 +00:00
|
|
|
#include "LyXView.h"
|
2003-09-05 10:55:42 +00:00
|
|
|
#include "metricsinfo.h"
|
2003-10-24 09:45:07 +00:00
|
|
|
#include "paragraph.h"
|
2003-09-05 10:55:42 +00:00
|
|
|
#include "rowpainter.h"
|
2002-06-25 15:59:10 +00:00
|
|
|
|
2006-08-17 22:00:07 +00:00
|
|
|
#include "gettext.h"
|
2002-06-25 15:59:10 +00:00
|
|
|
#include "support/filetools.h" // LibFileSearch
|
2006-07-08 22:06:50 +00:00
|
|
|
#include "support/forkedcontr.h"
|
2002-06-25 15:59:10 +00:00
|
|
|
|
|
|
|
#include <boost/utility.hpp>
|
|
|
|
#include <boost/bind.hpp>
|
2006-07-13 16:37:55 +00:00
|
|
|
#include <boost/current_function.hpp>
|
2002-06-25 15:59:10 +00:00
|
|
|
|
2006-04-08 22:31:11 +00:00
|
|
|
using lyx::support::libFileSearch;
|
2006-07-08 22:06:50 +00:00
|
|
|
using lyx::support::ForkedcallsController;
|
2003-06-30 23:56:22 +00:00
|
|
|
|
2003-10-17 18:01:15 +00:00
|
|
|
using std::endl;
|
2000-02-18 22:22:42 +00:00
|
|
|
using std::min;
|
2002-06-18 15:44:30 +00:00
|
|
|
using std::max;
|
2003-10-06 15:43:21 +00:00
|
|
|
using std::string;
|
2003-09-05 10:55:42 +00:00
|
|
|
|
2002-06-21 17:16:35 +00:00
|
|
|
|
2006-07-13 16:37:55 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// All the below connection objects are needed because of a bug in some
|
|
|
|
// versions of GCC (<=2.96 are on the suspects list.) By having and assigning
|
|
|
|
// to these connections we avoid a segfault upon startup, and also at exit.
|
|
|
|
// (Lgb)
|
|
|
|
|
|
|
|
boost::signals::connection timecon;
|
|
|
|
|
|
|
|
} // anon namespace
|
|
|
|
|
2006-10-23 08:47:27 +00:00
|
|
|
namespace lyx {
|
|
|
|
namespace frontend {
|
|
|
|
|
2006-08-15 21:57:23 +00:00
|
|
|
WorkArea::WorkArea(LyXView & lyx_view)
|
2006-10-23 08:47:27 +00:00
|
|
|
: buffer_view_(0), lyx_view_(lyx_view), greyed_out_(true),
|
|
|
|
cursor_visible_(false), cursor_timeout_(400)
|
1999-09-27 18:44:28 +00:00
|
|
|
{
|
2002-06-25 15:59:10 +00:00
|
|
|
// Start loading the pixmap as soon as possible
|
2006-10-23 08:47:27 +00:00
|
|
|
//if (lyxrc.show_banner) {
|
|
|
|
// showBanner();
|
|
|
|
//}
|
2006-07-08 22:06:50 +00:00
|
|
|
|
|
|
|
// Setup the signals
|
|
|
|
timecon = cursor_timeout_.timeout
|
|
|
|
.connect(boost::bind(&WorkArea::toggleCursor, this));
|
|
|
|
|
|
|
|
cursor_timeout_.start();
|
1999-09-27 18:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-06-26 16:55:35 +00:00
|
|
|
void WorkArea::setBufferView(BufferView * buffer_view)
|
|
|
|
{
|
2006-09-19 13:36:20 +00:00
|
|
|
if (buffer_view_) {
|
2006-08-24 14:10:22 +00:00
|
|
|
message_connection_.disconnect();
|
2006-09-19 13:36:20 +00:00
|
|
|
lyx_view_.disconnectBufferView();
|
|
|
|
}
|
2006-08-24 14:10:22 +00:00
|
|
|
|
2006-09-22 09:47:39 +00:00
|
|
|
theApp->setBufferView(buffer_view);
|
|
|
|
|
2006-08-16 15:24:38 +00:00
|
|
|
hideCursor();
|
2006-06-26 16:55:35 +00:00
|
|
|
buffer_view_ = buffer_view;
|
2006-08-16 15:24:38 +00:00
|
|
|
toggleCursor();
|
2006-08-24 14:10:22 +00:00
|
|
|
|
|
|
|
message_connection_ = buffer_view_->message.connect(
|
|
|
|
boost::bind(&WorkArea::displayMessage, this, _1));
|
2006-09-19 13:36:20 +00:00
|
|
|
|
|
|
|
lyx_view_.connectBufferView(*buffer_view);
|
2006-06-26 16:55:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-03 22:31:51 +00:00
|
|
|
BufferView & WorkArea::bufferView()
|
|
|
|
{
|
|
|
|
return *buffer_view_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BufferView const & WorkArea::bufferView() const
|
|
|
|
{
|
|
|
|
return *buffer_view_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-06-20 08:39:16 +00:00
|
|
|
void WorkArea::checkAndGreyOut()
|
2004-11-30 01:59:49 +00:00
|
|
|
{
|
|
|
|
if (greyed_out_)
|
|
|
|
greyOut();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-08 20:24:32 +00:00
|
|
|
void WorkArea::redraw()
|
2002-06-18 15:44:30 +00:00
|
|
|
{
|
2006-07-13 16:37:55 +00:00
|
|
|
if (!buffer_view_)
|
|
|
|
return;
|
2006-07-08 20:24:32 +00:00
|
|
|
|
|
|
|
if (!buffer_view_->buffer()) {
|
2006-10-23 08:47:27 +00:00
|
|
|
checkAndGreyOut();
|
2006-07-08 20:24:32 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-07-14 09:56:21 +00:00
|
|
|
buffer_view_->updateMetrics(false);
|
2006-08-16 15:24:38 +00:00
|
|
|
|
|
|
|
updateScrollbar();
|
|
|
|
|
2006-07-08 20:24:32 +00:00
|
|
|
ViewMetricsInfo const & vi = buffer_view_->viewMetricsInfo();
|
2006-07-13 16:37:55 +00:00
|
|
|
greyed_out_ = false;
|
2006-10-23 08:47:27 +00:00
|
|
|
|
2006-10-21 10:56:36 +00:00
|
|
|
lyxerr[Debug::WORKAREA] << "WorkArea::redraw screen" << endl;
|
2006-03-20 15:05:07 +00:00
|
|
|
int const ymin = std::max(vi.y1, 0);
|
2006-10-23 08:47:27 +00:00
|
|
|
int const ymax = vi.p2 < vi.size - 1 ? vi.y2 : height();
|
|
|
|
|
2006-06-20 08:39:16 +00:00
|
|
|
expose(0, ymin, width(), ymax - ymin);
|
2006-07-08 20:24:32 +00:00
|
|
|
|
2006-10-23 08:47:27 +00:00
|
|
|
//lyxerr[Debug::WORKAREA]
|
|
|
|
//<< " ymin = " << ymin << " width() = " << width()
|
|
|
|
// << " ymax-ymin = " << ymax-ymin << std::endl;
|
2006-10-21 11:29:34 +00:00
|
|
|
|
2006-10-23 08:47:27 +00:00
|
|
|
if (lyxerr.debugging(Debug::WORKAREA))
|
2006-10-21 11:29:34 +00:00
|
|
|
buffer_view_->coordCache().dump();
|
2002-06-18 15:44:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-07-13 16:37:55 +00:00
|
|
|
void WorkArea::processKeySym(LyXKeySymPtr key,
|
|
|
|
key_modifier::state state)
|
2006-07-08 21:04:21 +00:00
|
|
|
{
|
2006-07-08 22:06:50 +00:00
|
|
|
hideCursor();
|
2006-10-13 14:15:31 +00:00
|
|
|
|
|
|
|
theLyXFunc().setLyXView(&lyx_view_);
|
2006-10-11 17:24:46 +00:00
|
|
|
theLyXFunc().processKeySym(key, state);
|
2006-07-13 16:37:55 +00:00
|
|
|
|
2006-07-08 22:06:50 +00:00
|
|
|
/* This is perhaps a bit of a hack. When we move
|
|
|
|
* around, or type, it's nice to be able to see
|
|
|
|
* the cursor immediately after the keypress. So
|
|
|
|
* we reset the toggle timeout and force the visibility
|
|
|
|
* of the cursor. Note we cannot do this inside
|
|
|
|
* dispatch() itself, because that's called recursively.
|
|
|
|
*/
|
2006-09-16 14:00:13 +00:00
|
|
|
// if (buffer_view_->buffer())
|
2006-07-08 22:06:50 +00:00
|
|
|
toggleCursor();
|
2006-07-13 16:37:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void WorkArea::dispatch(FuncRequest const & cmd0)
|
|
|
|
{
|
2006-08-15 21:57:23 +00:00
|
|
|
// Handle drag&drop
|
|
|
|
if (cmd0.action == LFUN_FILE_OPEN) {
|
|
|
|
lyx_view_.dispatch(cmd0);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-10-12 15:23:49 +00:00
|
|
|
theLyXFunc().setLyXView(&lyx_view_);
|
|
|
|
|
2006-07-13 16:37:55 +00:00
|
|
|
buffer_view_->workAreaDispatch(cmd0);
|
2006-08-15 21:57:23 +00:00
|
|
|
|
|
|
|
// Skip these when selecting
|
|
|
|
if (cmd0.action != LFUN_MOUSE_MOTION) {
|
|
|
|
lyx_view_.updateLayoutChoice();
|
2006-09-26 10:06:46 +00:00
|
|
|
lyx_view_.updateMenubar();
|
2006-08-15 21:57:23 +00:00
|
|
|
lyx_view_.updateToolbars();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Slight hack: this is only called currently when we
|
|
|
|
// clicked somewhere, so we force through the display
|
|
|
|
// of the new status here.
|
|
|
|
lyx_view_.clearMessage();
|
|
|
|
|
2006-07-13 16:37:55 +00:00
|
|
|
redraw();
|
|
|
|
}
|
|
|
|
|
2006-07-08 22:06:50 +00:00
|
|
|
|
2006-07-13 16:37:55 +00:00
|
|
|
void WorkArea::resizeBufferView()
|
|
|
|
{
|
2006-08-17 22:00:07 +00:00
|
|
|
lyx_view_.busy(true);
|
2006-09-11 08:54:10 +00:00
|
|
|
lyx_view_.message(_("Formatting document..."));
|
2006-07-13 16:37:55 +00:00
|
|
|
buffer_view_->workAreaResize(width(), height());
|
2006-08-15 21:57:23 +00:00
|
|
|
lyx_view_.updateLayoutChoice();
|
|
|
|
redraw();
|
2006-08-17 22:00:07 +00:00
|
|
|
lyx_view_.busy(false);
|
|
|
|
lyx_view_.clearMessage();
|
2006-08-15 21:57:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-16 15:24:38 +00:00
|
|
|
void WorkArea::updateScrollbar()
|
|
|
|
{
|
|
|
|
buffer_view_->updateScrollbar();
|
|
|
|
ScrollbarParameters const & scroll_ = buffer_view_->scrollbarParameters();
|
|
|
|
setScrollbarParams(scroll_.height, scroll_.position,
|
|
|
|
scroll_.lineScrollHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-15 21:57:23 +00:00
|
|
|
void WorkArea::scrollBufferView(int position)
|
|
|
|
{
|
|
|
|
buffer_view_->scrollDocView(position);
|
2006-07-13 16:37:55 +00:00
|
|
|
redraw();
|
2006-08-16 15:24:38 +00:00
|
|
|
hideCursor();
|
|
|
|
if (lyxrc.cursor_follows_scrollbar) {
|
|
|
|
buffer_view_->setCursorFromScrollbar();
|
|
|
|
lyx_view_.updateLayoutChoice();
|
|
|
|
}
|
|
|
|
toggleCursor();
|
2006-07-08 21:04:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-06-20 08:39:16 +00:00
|
|
|
void WorkArea::greyOut()
|
2002-06-18 15:44:30 +00:00
|
|
|
{
|
2004-11-30 01:59:49 +00:00
|
|
|
greyed_out_ = true;
|
2006-10-23 08:47:27 +00:00
|
|
|
expose(4, 5, 3, 3);
|
2002-06-18 15:44:30 +00:00
|
|
|
}
|
2006-06-20 08:39:16 +00:00
|
|
|
|
2006-07-08 22:06:50 +00:00
|
|
|
|
|
|
|
void WorkArea::showCursor()
|
|
|
|
{
|
|
|
|
if (cursor_visible_)
|
|
|
|
return;
|
|
|
|
|
2006-09-16 14:00:13 +00:00
|
|
|
if (!buffer_view_->buffer())
|
2006-07-08 22:06:50 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
CursorShape shape = BAR_SHAPE;
|
|
|
|
|
|
|
|
LyXText const & text = *buffer_view_->getLyXText();
|
|
|
|
LyXFont const & realfont = text.real_current_font;
|
|
|
|
BufferParams const & bp = buffer_view_->buffer()->params();
|
|
|
|
bool const samelang = realfont.language() == bp.language;
|
|
|
|
bool const isrtl = realfont.isVisibleRightToLeft();
|
|
|
|
|
|
|
|
if (!samelang || isrtl != bp.language->rightToLeft()) {
|
|
|
|
shape = L_SHAPE;
|
|
|
|
if (isrtl)
|
|
|
|
shape = REVERSED_L_SHAPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The ERT language hack needs fixing up
|
|
|
|
if (realfont.language() == latex_language)
|
|
|
|
shape = BAR_SHAPE;
|
|
|
|
|
|
|
|
LyXFont const font = buffer_view_->cursor().getFont();
|
2006-10-11 17:24:46 +00:00
|
|
|
FontMetrics const & fm = theFontMetrics(font);
|
2006-10-07 16:15:06 +00:00
|
|
|
int const asc = fm.maxAscent();
|
|
|
|
int const des = fm.maxDescent();
|
2006-07-08 22:06:50 +00:00
|
|
|
int h = asc + des;
|
|
|
|
int x = 0;
|
|
|
|
int y = 0;
|
|
|
|
buffer_view_->cursor().getPos(x, y);
|
|
|
|
y -= asc;
|
|
|
|
|
|
|
|
// if it doesn't touch the screen, don't try to show it
|
|
|
|
if (y + h < 0 || y >= height())
|
|
|
|
return;
|
|
|
|
|
|
|
|
cursor_visible_ = true;
|
|
|
|
showCursor(x, y, h, shape);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void WorkArea::hideCursor()
|
|
|
|
{
|
|
|
|
if (!cursor_visible_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
cursor_visible_ = false;
|
|
|
|
removeCursor();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void WorkArea::toggleCursor()
|
|
|
|
{
|
|
|
|
if (buffer_view_->buffer()) {
|
|
|
|
|
|
|
|
if (cursor_visible_)
|
|
|
|
hideCursor();
|
|
|
|
else
|
|
|
|
showCursor();
|
|
|
|
|
|
|
|
// Use this opportunity to deal with any child processes that
|
|
|
|
// have finished but are waiting to communicate this fact
|
|
|
|
// to the rest of LyX.
|
|
|
|
ForkedcallsController & fcc = ForkedcallsController::get();
|
|
|
|
fcc.handleCompletedProcesses();
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor_timeout_.restart();
|
|
|
|
}
|
|
|
|
|
2006-08-24 14:10:22 +00:00
|
|
|
|
2006-09-11 08:54:10 +00:00
|
|
|
void WorkArea::displayMessage(lyx::docstring const & message)
|
2006-08-24 14:10:22 +00:00
|
|
|
{
|
|
|
|
lyx_view_.message(message);
|
|
|
|
}
|
|
|
|
|
2006-06-20 08:39:16 +00:00
|
|
|
} // namespace frontend
|
|
|
|
} // namespace lyx
|