3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
7 * \author Abdelrazak Younes
9 * Full author contact details are available in file CREDITS.
11 * Splash screen code added by Angus Leeming
18 #include "font_metrics.h"
19 #include "funcrequest.h"
24 #include "BufferView.h"
26 #include "bufferparams.h"
27 #include "coordcache.h"
37 #include "metricsinfo.h"
38 #include "paragraph.h"
39 #include "rowpainter.h"
42 #include "graphics/GraphicsImage.h"
43 #include "graphics/GraphicsLoader.h"
45 #include "support/filetools.h" // LibFileSearch
46 #include "support/forkedcontr.h"
48 #include <boost/utility.hpp>
49 #include <boost/bind.hpp>
50 #include <boost/current_function.hpp>
51 #include <boost/signals/trackable.hpp>
53 using lyx::support::libFileSearch;
54 using lyx::support::ForkedcallsController;
65 // FIXME: The SplashScreen should be transfered to the
66 // LyXView and create a WorkArea only when a new buffer exists. This
67 // will allow to call WorkArea::redraw() in the constructor.
68 class SplashScreen : boost::noncopyable, boost::signals::trackable {
70 /// This is a singleton class. Get the instance.
71 static SplashScreen const & get();
73 lyx::graphics::Image const * image() const { return loader_.image(); }
75 string const & text() const { return text_; }
77 LyXFont const & font() const { return font_; }
79 void connect(lyx::graphics::Loader::slot_type const & slot) const {
80 loader_.connect(slot);
83 void startLoading() const {
84 if (loader_.status() == lyx::graphics::WaitingToLoad)
85 loader_.startLoading();
89 /** Make the c-tor private so we can control how many objects
95 lyx::graphics::Loader loader_;
96 /// The text to be written on top of the pixmap
103 SplashScreen const & SplashScreen::get()
105 static SplashScreen singleton;
110 SplashScreen::SplashScreen()
111 : text_(lyx_version ? lyx_version : "unknown")
113 if (!lyxrc.show_banner)
116 string const file = libFileSearch("images", "banner", "ppm");
120 // The font used to display the version info
121 font_.setFamily(LyXFont::SANS_FAMILY);
122 font_.setSeries(LyXFont::BOLD_SERIES);
123 font_.setSize(LyXFont::SIZE_NORMAL);
124 font_.setColor(LColor::yellow);
126 // Load up the graphics file
132 // All the below connection objects are needed because of a bug in some
133 // versions of GCC (<=2.96 are on the suspects list.) By having and assigning
134 // to these connections we avoid a segfault upon startup, and also at exit.
137 boost::signals::connection timecon;
141 WorkArea::WorkArea(LyXView & lyx_view)
142 : buffer_view_(0), lyx_view_(lyx_view), greyed_out_(true),
143 cursor_visible_(false), cursor_timeout_(400)
145 // Start loading the pixmap as soon as possible
146 if (lyxrc.show_banner) {
147 SplashScreen const & splash = SplashScreen::get();
148 splash.connect(boost::bind(&WorkArea::checkAndGreyOut, this));
149 splash.startLoading();
153 timecon = cursor_timeout_.timeout
154 .connect(boost::bind(&WorkArea::toggleCursor, this));
156 cursor_timeout_.start();
160 void WorkArea::setBufferView(BufferView * buffer_view)
162 buffer_view_ = buffer_view;
166 BufferView & WorkArea::bufferView()
168 return *buffer_view_;
172 BufferView const & WorkArea::bufferView() const
174 return *buffer_view_;
178 void WorkArea::checkAndGreyOut()
185 void WorkArea::redraw()
190 if (!buffer_view_->buffer()) {
195 buffer_view_->updateMetrics(false);
196 ViewMetricsInfo const & vi = buffer_view_->viewMetricsInfo();
198 getPainter().start();
199 paintText(*buffer_view_, vi, getPainter());
200 lyxerr[Debug::DEBUG] << "Redraw screen" << endl;
201 int const ymin = std::max(vi.y1, 0);
203 ( vi.p2 < vi.size - 1 ? vi.y2 : height() );
204 expose(0, ymin, width(), ymax - ymin);
208 << " ymin = " << ymin << " width() = " << width()
209 << " ymax-ymin = " << ymax-ymin << std::endl;
213 void WorkArea::processKeySym(LyXKeySymPtr key,
214 key_modifier::state state)
217 lyx_view_.getLyXFunc().processKeySym(key, state);
219 /* This is perhaps a bit of a hack. When we move
220 * around, or type, it's nice to be able to see
221 * the cursor immediately after the keypress. So
222 * we reset the toggle timeout and force the visibility
223 * of the cursor. Note we cannot do this inside
224 * dispatch() itself, because that's called recursively.
226 // if (buffer_view_->available())
229 // uneeded "redraw()" call commented out for now.
230 // When/if the call to LyXView::redrawWorkArea() in "lyxfunc.C:1610"
231 // is not needed anymore, this line should be uncommented out
236 void WorkArea::dispatch(FuncRequest const & cmd0)
239 if (cmd0.action == LFUN_FILE_OPEN) {
240 lyx_view_.dispatch(cmd0);
244 buffer_view_->workAreaDispatch(cmd0);
246 // Skip these when selecting
247 if (cmd0.action != LFUN_MOUSE_MOTION) {
248 lyx_view_.updateLayoutChoice();
249 lyx_view_.updateToolbars();
252 // Slight hack: this is only called currently when we
253 // clicked somewhere, so we force through the display
254 // of the new status here.
255 lyx_view_.clearMessage();
261 void WorkArea::resizeBufferView()
263 buffer_view_->workAreaResize(width(), height());
264 lyx_view_.updateLayoutChoice();
269 void WorkArea::scrollBufferView(int position)
271 buffer_view_->scrollDocView(position);
272 lyx_view_.updateLayoutChoice();
277 void WorkArea::greyOut()
280 getPainter().start();
282 getPainter().fillRectangle(0, 0,
287 // Add a splash screen to the centre of the work area
288 SplashScreen const & splash = SplashScreen::get();
289 lyx::graphics::Image const * const splash_image = splash.image();
291 int const w = splash_image->getWidth();
292 int const h = splash_image->getHeight();
294 int x = (width() - w) / 2;
295 int y = (height() - h) / 2;
297 getPainter().image(x, y, w, h, *splash_image);
302 string stext = splash.text();
303 docstring dstext(stext.begin(), stext.end());
304 getPainter().text(x, y, dstext, splash.font());
306 expose(0, 0, width(), height());
311 void WorkArea::showCursor()
316 if (!buffer_view_->available())
319 CursorShape shape = BAR_SHAPE;
321 LyXText const & text = *buffer_view_->getLyXText();
322 LyXFont const & realfont = text.real_current_font;
323 BufferParams const & bp = buffer_view_->buffer()->params();
324 bool const samelang = realfont.language() == bp.language;
325 bool const isrtl = realfont.isVisibleRightToLeft();
327 if (!samelang || isrtl != bp.language->rightToLeft()) {
330 shape = REVERSED_L_SHAPE;
333 // The ERT language hack needs fixing up
334 if (realfont.language() == latex_language)
337 LyXFont const font = buffer_view_->cursor().getFont();
338 int const asc = font_metrics::maxAscent(font);
339 int const des = font_metrics::maxDescent(font);
343 buffer_view_->cursor().getPos(x, y);
346 // if it doesn't touch the screen, don't try to show it
347 if (y + h < 0 || y >= height())
350 cursor_visible_ = true;
351 showCursor(x, y, h, shape);
355 void WorkArea::hideCursor()
357 if (!cursor_visible_)
360 cursor_visible_ = false;
365 void WorkArea::toggleCursor()
367 if (buffer_view_->buffer()) {
374 // Use this opportunity to deal with any child processes that
375 // have finished but are waiting to communicate this fact
376 // to the rest of LyX.
377 ForkedcallsController & fcc = ForkedcallsController::get();
378 fcc.handleCompletedProcesses();
381 cursor_timeout_.restart();
384 } // namespace frontend