mirror of
https://git.lyx.org/repos/lyx.git
synced 2024-11-08 20:32:49 +00:00
3f0fadc48a
Most of qt3 and gtk should stay compilable except for LyXView.h because the LyXFunc instance has been transferred to Application. * frontends/Application: new class aimed to be the one unique interface between the frontend and the kernel. Contains one global pointer to the unique instanciation theApp. * frontends/qt4/GuiApplication: renamed from qt4/Application, the qt4 specialisation of the Application class. Contains one global pointer to the unique instanciation guiApp (equal to theApp but pointing to a GuiApplication instead). * frontends/qt4/lyx_gui.C: most of the code has been moved to Application and GuiApplication All other file: adapted to new API. git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@15114 a592a061-630c-0410-9148-cb99ea01b6c8
428 lines
9.2 KiB
C
428 lines
9.2 KiB
C
/**
|
|
* \file WorkArea.C
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author John Levon
|
|
* \author Abdelrazak Younes
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*
|
|
* Splash screen code added by Angus Leeming
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "WorkArea.h"
|
|
|
|
#include "font_metrics.h"
|
|
#include "funcrequest.h"
|
|
#include "lyx_gui.h"
|
|
#include "lyxfunc.h"
|
|
#include "Painter.h"
|
|
|
|
#include "BufferView.h"
|
|
#include "buffer.h"
|
|
#include "bufferparams.h"
|
|
#include "coordcache.h"
|
|
#include "cursor.h"
|
|
#include "debug.h"
|
|
#include "language.h"
|
|
#include "LColor.h"
|
|
#include "lyxfont.h"
|
|
#include "lyxrc.h"
|
|
#include "lyxrow.h"
|
|
#include "lyxtext.h"
|
|
#include "LyXView.h"
|
|
#include "metricsinfo.h"
|
|
#include "paragraph.h"
|
|
#include "rowpainter.h"
|
|
#include "version.h"
|
|
|
|
#include "graphics/GraphicsImage.h"
|
|
#include "graphics/GraphicsLoader.h"
|
|
|
|
#include "gettext.h"
|
|
#include "support/filetools.h" // LibFileSearch
|
|
#include "support/forkedcontr.h"
|
|
|
|
#include <boost/utility.hpp>
|
|
#include <boost/bind.hpp>
|
|
#include <boost/current_function.hpp>
|
|
|
|
using lyx::support::libFileSearch;
|
|
using lyx::support::ForkedcallsController;
|
|
|
|
using std::endl;
|
|
using std::min;
|
|
using std::max;
|
|
using std::string;
|
|
|
|
|
|
namespace lyx {
|
|
namespace frontend {
|
|
|
|
// FIXME: The SplashScreen should be transfered to the
|
|
// LyXView and create a WorkArea only when a new buffer exists. This
|
|
// will allow to call WorkArea::redraw() in the constructor.
|
|
class SplashScreen : boost::noncopyable, boost::signals::trackable {
|
|
public:
|
|
/// This is a singleton class. Get the instance.
|
|
static SplashScreen const & get();
|
|
///
|
|
lyx::graphics::Image const * image() const { return loader_.image(); }
|
|
///
|
|
string const & text() const { return text_; }
|
|
///
|
|
LyXFont const & font() const { return font_; }
|
|
///
|
|
void connect(lyx::graphics::Loader::slot_type const & slot) const {
|
|
loader_.connect(slot);
|
|
}
|
|
///
|
|
void startLoading() const {
|
|
if (loader_.status() == lyx::graphics::WaitingToLoad)
|
|
loader_.startLoading();
|
|
}
|
|
|
|
private:
|
|
/** Make the c-tor private so we can control how many objects
|
|
* are instantiated.
|
|
*/
|
|
SplashScreen();
|
|
|
|
///
|
|
lyx::graphics::Loader loader_;
|
|
/// The text to be written on top of the pixmap
|
|
string const text_;
|
|
/// in this font...
|
|
LyXFont font_;
|
|
};
|
|
|
|
|
|
SplashScreen const & SplashScreen::get()
|
|
{
|
|
static SplashScreen singleton;
|
|
return singleton;
|
|
}
|
|
|
|
|
|
SplashScreen::SplashScreen()
|
|
: text_(lyx_version ? lyx_version : "unknown")
|
|
{
|
|
if (!lyxrc.show_banner)
|
|
return;
|
|
|
|
string const file = libFileSearch("images", "banner", "ppm");
|
|
if (file.empty())
|
|
return;
|
|
|
|
// The font used to display the version info
|
|
font_.setFamily(LyXFont::SANS_FAMILY);
|
|
font_.setSeries(LyXFont::BOLD_SERIES);
|
|
font_.setSize(LyXFont::SIZE_NORMAL);
|
|
font_.setColor(LColor::yellow);
|
|
|
|
// Load up the graphics file
|
|
loader_.reset(file);
|
|
}
|
|
|
|
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
|
|
|
|
WorkArea::WorkArea(LyXView & lyx_view)
|
|
: buffer_view_(0), lyx_view_(lyx_view), greyed_out_(true),
|
|
cursor_visible_(false), cursor_timeout_(400)
|
|
{
|
|
// Start loading the pixmap as soon as possible
|
|
if (lyxrc.show_banner) {
|
|
SplashScreen const & splash = SplashScreen::get();
|
|
splash.connect(boost::bind(&WorkArea::checkAndGreyOut, this));
|
|
splash.startLoading();
|
|
}
|
|
|
|
// Setup the signals
|
|
timecon = cursor_timeout_.timeout
|
|
.connect(boost::bind(&WorkArea::toggleCursor, this));
|
|
|
|
cursor_timeout_.start();
|
|
}
|
|
|
|
|
|
void WorkArea::setBufferView(BufferView * buffer_view)
|
|
{
|
|
if (buffer_view_) {
|
|
message_connection_.disconnect();
|
|
lyx_view_.disconnectBufferView();
|
|
}
|
|
|
|
theApp->setBufferView(buffer_view);
|
|
|
|
hideCursor();
|
|
buffer_view_ = buffer_view;
|
|
toggleCursor();
|
|
|
|
message_connection_ = buffer_view_->message.connect(
|
|
boost::bind(&WorkArea::displayMessage, this, _1));
|
|
|
|
lyx_view_.connectBufferView(*buffer_view);
|
|
}
|
|
|
|
|
|
BufferView & WorkArea::bufferView()
|
|
{
|
|
return *buffer_view_;
|
|
}
|
|
|
|
|
|
BufferView const & WorkArea::bufferView() const
|
|
{
|
|
return *buffer_view_;
|
|
}
|
|
|
|
|
|
void WorkArea::checkAndGreyOut()
|
|
{
|
|
if (greyed_out_)
|
|
greyOut();
|
|
}
|
|
|
|
|
|
void WorkArea::redraw()
|
|
{
|
|
if (!buffer_view_)
|
|
return;
|
|
|
|
if (!buffer_view_->buffer()) {
|
|
greyOut();
|
|
updateScrollbar();
|
|
return;
|
|
}
|
|
|
|
buffer_view_->updateMetrics(false);
|
|
|
|
updateScrollbar();
|
|
|
|
ViewMetricsInfo const & vi = buffer_view_->viewMetricsInfo();
|
|
greyed_out_ = false;
|
|
getPainter().start();
|
|
paintText(*buffer_view_, vi, getPainter());
|
|
lyxerr[Debug::DEBUG] << "Redraw screen" << endl;
|
|
int const ymin = std::max(vi.y1, 0);
|
|
int const ymax =
|
|
( vi.p2 < vi.size - 1 ? vi.y2 : height() );
|
|
expose(0, ymin, width(), ymax - ymin);
|
|
getPainter().end();
|
|
|
|
lyxerr[Debug::DEBUG]
|
|
<< " ymin = " << ymin << " width() = " << width()
|
|
<< " ymax-ymin = " << ymax-ymin << std::endl;
|
|
}
|
|
|
|
|
|
void WorkArea::processKeySym(LyXKeySymPtr key,
|
|
key_modifier::state state)
|
|
{
|
|
hideCursor();
|
|
lyx_view_.getLyXFunc().processKeySym(key, state);
|
|
|
|
/* 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.
|
|
*/
|
|
// if (buffer_view_->buffer())
|
|
toggleCursor();
|
|
|
|
// uneeded "redraw()" call commented out for now.
|
|
// When/if the call to LyXView::redrawWorkArea() in "lyxfunc.C:1610"
|
|
// is not needed anymore, this line should be uncommented out
|
|
//redraw();
|
|
}
|
|
|
|
|
|
void WorkArea::dispatch(FuncRequest const & cmd0)
|
|
{
|
|
// Handle drag&drop
|
|
if (cmd0.action == LFUN_FILE_OPEN) {
|
|
lyx_view_.dispatch(cmd0);
|
|
return;
|
|
}
|
|
|
|
buffer_view_->workAreaDispatch(cmd0);
|
|
|
|
// Skip these when selecting
|
|
if (cmd0.action != LFUN_MOUSE_MOTION) {
|
|
lyx_view_.updateLayoutChoice();
|
|
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();
|
|
|
|
redraw();
|
|
}
|
|
|
|
|
|
void WorkArea::resizeBufferView()
|
|
{
|
|
lyx_view_.busy(true);
|
|
lyx_view_.message(_("Formatting document..."));
|
|
buffer_view_->workAreaResize(width(), height());
|
|
lyx_view_.updateLayoutChoice();
|
|
redraw();
|
|
lyx_view_.busy(false);
|
|
lyx_view_.clearMessage();
|
|
}
|
|
|
|
|
|
void WorkArea::updateScrollbar()
|
|
{
|
|
buffer_view_->updateScrollbar();
|
|
ScrollbarParameters const & scroll_ = buffer_view_->scrollbarParameters();
|
|
setScrollbarParams(scroll_.height, scroll_.position,
|
|
scroll_.lineScrollHeight);
|
|
}
|
|
|
|
|
|
void WorkArea::scrollBufferView(int position)
|
|
{
|
|
buffer_view_->scrollDocView(position);
|
|
redraw();
|
|
hideCursor();
|
|
if (lyxrc.cursor_follows_scrollbar) {
|
|
buffer_view_->setCursorFromScrollbar();
|
|
lyx_view_.updateLayoutChoice();
|
|
}
|
|
toggleCursor();
|
|
}
|
|
|
|
|
|
void WorkArea::greyOut()
|
|
{
|
|
greyed_out_ = true;
|
|
getPainter().start();
|
|
|
|
getPainter().fillRectangle(0, 0,
|
|
width(),
|
|
height(),
|
|
LColor::bottomarea);
|
|
|
|
// Add a splash screen to the centre of the work area
|
|
SplashScreen const & splash = SplashScreen::get();
|
|
lyx::graphics::Image const * const splash_image = splash.image();
|
|
if (splash_image) {
|
|
int const w = splash_image->getWidth();
|
|
int const h = splash_image->getHeight();
|
|
|
|
int x = (width() - w) / 2;
|
|
int y = (height() - h) / 2;
|
|
|
|
getPainter().image(x, y, w, h, *splash_image);
|
|
|
|
x += 260;
|
|
y += 265;
|
|
|
|
string stext = splash.text();
|
|
docstring dstext(stext.begin(), stext.end());
|
|
getPainter().text(x, y, dstext, splash.font());
|
|
}
|
|
expose(0, 0, width(), height());
|
|
getPainter().end();
|
|
}
|
|
|
|
|
|
void WorkArea::showCursor()
|
|
{
|
|
if (cursor_visible_)
|
|
return;
|
|
|
|
if (!buffer_view_->buffer())
|
|
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();
|
|
int const asc = font_metrics::maxAscent(font);
|
|
int const des = font_metrics::maxDescent(font);
|
|
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();
|
|
}
|
|
|
|
|
|
void WorkArea::displayMessage(lyx::docstring const & message)
|
|
{
|
|
lyx_view_.message(message);
|
|
}
|
|
|
|
} // namespace frontend
|
|
} // namespace lyx
|