mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-14 04:21:56 +00:00
The BufferView/WorkArea/LyXView reorg a.k.a Multiple WorkAreas:
* Buffer: - get rid of cursor_ and anchor_ - ~Buffer(): update the labels of its master buffer before closing - closing(): pass the Buffer address. * BufferView(): - BufferView(): needs a valid Buffer (should be const in the future. - most of the change is about removing all test of buffer_ nullity. - resize(): deleted. - setBuffer(): deleted. * Application: - newLyXView(): simplification - updated design description in Application.h * Gui/GuiImplementation: remove all WorkAreas and BufferView creation/Deletion. Workareas are directly handled by LyXView/GuiView and BufferView is created/delete by WorkArea. * LyXView/GuiView: implement the new design What is not working yet: - the close tab button: it is implemented but does not show up. git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@19686 a592a061-630c-0410-9148-cb99ea01b6c8
This commit is contained in:
parent
7206f23872
commit
1356543c45
@ -22,6 +22,7 @@
|
|||||||
#include "Bullet.h"
|
#include "Bullet.h"
|
||||||
#include "Chktex.h"
|
#include "Chktex.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
|
#include "DocIterator.h"
|
||||||
#include "Encoding.h"
|
#include "Encoding.h"
|
||||||
#include "ErrorList.h"
|
#include "ErrorList.h"
|
||||||
#include "Exporter.h"
|
#include "Exporter.h"
|
||||||
@ -227,7 +228,12 @@ Buffer::~Buffer()
|
|||||||
// here the buffer should take care that it is
|
// here the buffer should take care that it is
|
||||||
// saved properly, before it goes into the void.
|
// saved properly, before it goes into the void.
|
||||||
|
|
||||||
closing();
|
Buffer * master = getMasterBuffer();
|
||||||
|
if (master != this && use_gui)
|
||||||
|
// We are closing buf which was a child document so we
|
||||||
|
// must update the labels and section numbering of its master
|
||||||
|
// Buffer.
|
||||||
|
updateLabels(*master);
|
||||||
|
|
||||||
if (!temppath().empty() && !destroyDir(FileName(temppath()))) {
|
if (!temppath().empty() && !destroyDir(FileName(temppath()))) {
|
||||||
Alert::warning(_("Could not remove temporary directory"),
|
Alert::warning(_("Could not remove temporary directory"),
|
||||||
@ -237,6 +243,8 @@ Buffer::~Buffer()
|
|||||||
|
|
||||||
// Remove any previewed LaTeX snippets associated with this buffer.
|
// Remove any previewed LaTeX snippets associated with this buffer.
|
||||||
graphics::Previews::get().removeLoader(*this);
|
graphics::Previews::get().removeLoader(*this);
|
||||||
|
|
||||||
|
closing(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1707,13 +1715,6 @@ void Buffer::buildMacros()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Buffer::saveCursor(StableDocIterator cur, StableDocIterator anc)
|
|
||||||
{
|
|
||||||
cursor_ = cur;
|
|
||||||
anchor_ = anc;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void Buffer::changeRefsIfUnique(docstring const & from, docstring const & to,
|
void Buffer::changeRefsIfUnique(docstring const & from, docstring const & to,
|
||||||
Inset::Code code)
|
Inset::Code code)
|
||||||
{
|
{
|
||||||
|
15
src/Buffer.h
15
src/Buffer.h
@ -25,7 +25,6 @@
|
|||||||
|
|
||||||
#include <iosfwd>
|
#include <iosfwd>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <map>
|
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
@ -50,7 +49,6 @@ class OutputParams;
|
|||||||
class ParConstIterator;
|
class ParConstIterator;
|
||||||
class ParIterator;
|
class ParIterator;
|
||||||
class ParagraphList;
|
class ParagraphList;
|
||||||
class StableDocIterator;
|
|
||||||
class TeXErrors;
|
class TeXErrors;
|
||||||
class TexRow;
|
class TexRow;
|
||||||
class TocBackend;
|
class TocBackend;
|
||||||
@ -159,7 +157,7 @@ public:
|
|||||||
/// Reset autosave timers for all users.
|
/// Reset autosave timers for all users.
|
||||||
boost::signal<void()> resetAutosaveTimers;
|
boost::signal<void()> resetAutosaveTimers;
|
||||||
/// This signal is emitting if the buffer is being closed.
|
/// This signal is emitting if the buffer is being closed.
|
||||||
boost::signal<void()> closing;
|
boost::signal<void(Buffer *)> closing;
|
||||||
|
|
||||||
|
|
||||||
/** Save file.
|
/** Save file.
|
||||||
@ -381,12 +379,6 @@ public:
|
|||||||
///
|
///
|
||||||
void insertMacro(docstring const & name, MacroData const & data);
|
void insertMacro(docstring const & name, MacroData const & data);
|
||||||
|
|
||||||
///
|
|
||||||
void saveCursor(StableDocIterator cursor, StableDocIterator anchor);
|
|
||||||
///
|
|
||||||
StableDocIterator getCursor() const { return cursor_; }
|
|
||||||
///
|
|
||||||
StableDocIterator getAnchor() const { return anchor_; }
|
|
||||||
///
|
///
|
||||||
void changeRefsIfUnique(docstring const & from, docstring const & to,
|
void changeRefsIfUnique(docstring const & from, docstring const & to,
|
||||||
Inset::Code code);
|
Inset::Code code);
|
||||||
@ -417,11 +409,6 @@ private:
|
|||||||
/// The pointer never changes although *pimpl_'s contents may.
|
/// The pointer never changes although *pimpl_'s contents may.
|
||||||
boost::scoped_ptr<Impl> const pimpl_;
|
boost::scoped_ptr<Impl> const pimpl_;
|
||||||
|
|
||||||
/// Save the cursor Position on Buffer switch
|
|
||||||
/// this would not be needed if every Buffer would have
|
|
||||||
/// it's BufferView, this should be FIXED in future.
|
|
||||||
StableDocIterator cursor_;
|
|
||||||
StableDocIterator anchor_;
|
|
||||||
/// A cache for the bibfiles (including bibfiles of loaded child
|
/// A cache for the bibfiles (including bibfiles of loaded child
|
||||||
/// documents), needed for appropriate update of natbib labels.
|
/// documents), needed for appropriate update of natbib labels.
|
||||||
mutable std::vector<support::FileName> bibfilesCache_;
|
mutable std::vector<support::FileName> bibfilesCache_;
|
||||||
|
@ -122,14 +122,21 @@ T * getInsetByCode(Cursor & cur, Inset::Code code)
|
|||||||
} // anon namespace
|
} // anon namespace
|
||||||
|
|
||||||
|
|
||||||
BufferView::BufferView()
|
BufferView::BufferView(Buffer & buf)
|
||||||
: width_(0), height_(0), buffer_(0), wh_(0),
|
: width_(0), height_(0), buffer_(buf), wh_(0),
|
||||||
cursor_(*this),
|
cursor_(*this),
|
||||||
multiparsel_cache_(false), anchor_ref_(0), offset_ref_(0),
|
multiparsel_cache_(false), anchor_ref_(0), offset_ref_(0),
|
||||||
intl_(new Intl), last_inset_(0)
|
intl_(new Intl), last_inset_(0)
|
||||||
{
|
{
|
||||||
xsel_cache_.set = false;
|
xsel_cache_.set = false;
|
||||||
intl_->initKeyMapper(lyxrc.use_kbmap);
|
intl_->initKeyMapper(lyxrc.use_kbmap);
|
||||||
|
|
||||||
|
cursor_.push(buffer_.inset());
|
||||||
|
cursor_.resetAnchor();
|
||||||
|
buffer_.text().setCurrentFont(cursor_);
|
||||||
|
|
||||||
|
if (graphics::Previews::status() != LyXRC::PREVIEW_OFF)
|
||||||
|
graphics::Previews::get().generateBufferPreviews(buffer_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -138,124 +145,15 @@ BufferView::~BufferView()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Buffer * BufferView::buffer() const
|
Buffer * BufferView::buffer()
|
||||||
{
|
{
|
||||||
return buffer_;
|
return &buffer_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Buffer * BufferView::setBuffer(Buffer * b)
|
Buffer const * BufferView::buffer() const
|
||||||
{
|
{
|
||||||
LYXERR(Debug::INFO) << BOOST_CURRENT_FUNCTION
|
return &buffer_;
|
||||||
<< "[ b = " << b << "]" << endl;
|
|
||||||
|
|
||||||
if (buffer_) {
|
|
||||||
// Save the current selection if any
|
|
||||||
cap::saveSelection(cursor_);
|
|
||||||
// Save the actual cursor position and anchor inside the
|
|
||||||
// buffer so that it can be restored in case we rechange
|
|
||||||
// to this buffer later on.
|
|
||||||
buffer_->saveCursor(cursor_.selectionBegin(),
|
|
||||||
cursor_.selectionEnd());
|
|
||||||
// update bookmark pit of the current buffer before switch
|
|
||||||
for (size_t i = 0; i < LyX::ref().session().bookmarks().size(); ++i) {
|
|
||||||
BookmarksSection::Bookmark const & bm = LyX::ref().session().bookmarks().bookmark(i);
|
|
||||||
if (buffer()->fileName() != bm.filename.absFilename())
|
|
||||||
continue;
|
|
||||||
// if top_id or bottom_pit, bottom_pos has been changed, update bookmark
|
|
||||||
// see http://bugzilla.lyx.org/show_bug.cgi?id=3092
|
|
||||||
pit_type new_pit;
|
|
||||||
pos_type new_pos;
|
|
||||||
int new_id;
|
|
||||||
boost::tie(new_pit, new_pos, new_id) = moveToPosition(bm.bottom_pit, bm.bottom_pos, bm.top_id, bm.top_pos);
|
|
||||||
if (bm.bottom_pit != new_pit || bm.bottom_pos != new_pos || bm.top_id != new_id )
|
|
||||||
const_cast<BookmarksSection::Bookmark &>(bm).updatePos(new_pit, new_pos, new_id);
|
|
||||||
}
|
|
||||||
// current buffer is going to be switched-off, save cursor pos
|
|
||||||
// Ideally, the whole cursor stack should be saved, but session
|
|
||||||
// currently can only handle bottom (whole document) level pit and pos.
|
|
||||||
// That is to say, if a cursor is in a nested inset, it will be
|
|
||||||
// restore to the left of the top level inset.
|
|
||||||
LyX::ref().session().lastFilePos().save(FileName(buffer_->fileName()),
|
|
||||||
boost::tie(cursor_.bottom().pit(), cursor_.bottom().pos()) );
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we're quitting lyx, don't bother updating stuff
|
|
||||||
if (quitting) {
|
|
||||||
buffer_ = 0;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
//FIXME Fix for bug 3440 is here.
|
|
||||||
// If we are closing current buffer, switch to the first in
|
|
||||||
// buffer list.
|
|
||||||
if (!b) {
|
|
||||||
LYXERR(Debug::INFO) << BOOST_CURRENT_FUNCTION
|
|
||||||
<< " No Buffer!" << endl;
|
|
||||||
// We are closing the buffer, use the first buffer as current
|
|
||||||
//FIXME 3440
|
|
||||||
// if (last_buffer_) buffer_ = last_buffer_;
|
|
||||||
// also check that this is in theBufferList()?
|
|
||||||
buffer_ = theBufferList().first();
|
|
||||||
} else {
|
|
||||||
//FIXME 3440
|
|
||||||
// last_buffer = buffer_;
|
|
||||||
// Set current buffer
|
|
||||||
buffer_ = b;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reset old cursor
|
|
||||||
cursor_ = Cursor(*this);
|
|
||||||
anchor_ref_ = 0;
|
|
||||||
offset_ref_ = 0;
|
|
||||||
|
|
||||||
if (!buffer_)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
LYXERR(Debug::INFO) << BOOST_CURRENT_FUNCTION
|
|
||||||
<< "Buffer addr: " << buffer_ << endl;
|
|
||||||
cursor_.push(buffer_->inset());
|
|
||||||
cursor_.resetAnchor();
|
|
||||||
buffer_->text().setCurrentFont(cursor_);
|
|
||||||
|
|
||||||
// Update the metrics now that we have a proper Cursor.
|
|
||||||
updateMetrics(false);
|
|
||||||
|
|
||||||
// FIXME: This code won't be needed once we switch to
|
|
||||||
// "one Buffer" / "one BufferView".
|
|
||||||
if (buffer_->getCursor().size() > 0 &&
|
|
||||||
buffer_->getAnchor().size() > 0)
|
|
||||||
{
|
|
||||||
cursor_.setCursor(buffer_->getAnchor().asDocIterator(&(buffer_->inset())));
|
|
||||||
cursor_.resetAnchor();
|
|
||||||
cursor_.setCursor(buffer_->getCursor().asDocIterator(&(buffer_->inset())));
|
|
||||||
cursor_.setSelection();
|
|
||||||
// do not set selection to the new buffer because we
|
|
||||||
// only paste recent selection.
|
|
||||||
|
|
||||||
// Make sure that the restored cursor is not broken. This can happen for
|
|
||||||
// example if this Buffer has been modified by another view.
|
|
||||||
cursor_.fixIfBroken();
|
|
||||||
|
|
||||||
if (fitCursor())
|
|
||||||
// Update the metrics if the cursor new position was off screen.
|
|
||||||
updateMetrics(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (graphics::Previews::status() != LyXRC::PREVIEW_OFF)
|
|
||||||
graphics::Previews::get().generateBufferPreviews(*buffer_);
|
|
||||||
return buffer_;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void BufferView::resize()
|
|
||||||
{
|
|
||||||
if (!buffer_)
|
|
||||||
return;
|
|
||||||
|
|
||||||
LYXERR(Debug::DEBUG) << BOOST_CURRENT_FUNCTION << endl;
|
|
||||||
|
|
||||||
updateMetrics(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -298,17 +196,11 @@ bool BufferView::update(Update::flags flags)
|
|||||||
<< "[fitcursor = " << (flags & Update::FitCursor)
|
<< "[fitcursor = " << (flags & Update::FitCursor)
|
||||||
<< ", forceupdate = " << (flags & Update::Force)
|
<< ", forceupdate = " << (flags & Update::Force)
|
||||||
<< ", singlepar = " << (flags & Update::SinglePar)
|
<< ", singlepar = " << (flags & Update::SinglePar)
|
||||||
<< "] buffer: " << buffer_ << endl;
|
<< "] buffer: " << &buffer_ << endl;
|
||||||
|
|
||||||
// Check needed to survive LyX startup
|
|
||||||
if (!buffer_)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
LYXERR(Debug::WORKAREA) << "BufferView::update" << std::endl;
|
|
||||||
|
|
||||||
// Update macro store
|
// Update macro store
|
||||||
if (!(cursor().inMathed() && cursor().inMacroMode()))
|
if (!(cursor().inMathed() && cursor().inMacroMode()))
|
||||||
buffer_->buildMacros();
|
buffer_.buildMacros();
|
||||||
|
|
||||||
// Now do the first drawing step if needed. This consists on updating
|
// Now do the first drawing step if needed. This consists on updating
|
||||||
// the CoordCache in updateMetrics().
|
// the CoordCache in updateMetrics().
|
||||||
@ -360,14 +252,7 @@ bool BufferView::update(Update::flags flags)
|
|||||||
|
|
||||||
void BufferView::updateScrollbar()
|
void BufferView::updateScrollbar()
|
||||||
{
|
{
|
||||||
if (!buffer_) {
|
Text & t = buffer_.text();
|
||||||
LYXERR(Debug::DEBUG) << BOOST_CURRENT_FUNCTION
|
|
||||||
<< " no text in updateScrollbar" << endl;
|
|
||||||
scrollbarParameters_.reset();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Text & t = buffer_->text();
|
|
||||||
TextMetrics & tm = text_metrics_[&t];
|
TextMetrics & tm = text_metrics_[&t];
|
||||||
|
|
||||||
int const parsize = int(t.paragraphs().size() - 1);
|
int const parsize = int(t.paragraphs().size() - 1);
|
||||||
@ -432,10 +317,7 @@ void BufferView::scrollDocView(int value)
|
|||||||
LYXERR(Debug::GUI) << BOOST_CURRENT_FUNCTION
|
LYXERR(Debug::GUI) << BOOST_CURRENT_FUNCTION
|
||||||
<< "[ value = " << value << "]" << endl;
|
<< "[ value = " << value << "]" << endl;
|
||||||
|
|
||||||
if (!buffer_)
|
Text & t = buffer_.text();
|
||||||
return;
|
|
||||||
|
|
||||||
Text & t = buffer_->text();
|
|
||||||
TextMetrics & tm = text_metrics_[&t];
|
TextMetrics & tm = text_metrics_[&t];
|
||||||
|
|
||||||
float const bar = value / float(wh_ * t.paragraphs().size());
|
float const bar = value / float(wh_ * t.paragraphs().size());
|
||||||
@ -453,10 +335,7 @@ void BufferView::scrollDocView(int value)
|
|||||||
|
|
||||||
void BufferView::setCursorFromScrollbar()
|
void BufferView::setCursorFromScrollbar()
|
||||||
{
|
{
|
||||||
if (!buffer_)
|
Text & t = buffer_.text();
|
||||||
return;
|
|
||||||
|
|
||||||
Text & t = buffer_->text();
|
|
||||||
|
|
||||||
int const height = 2 * defaultRowHeight();
|
int const height = 2 * defaultRowHeight();
|
||||||
int const first = height;
|
int const first = height;
|
||||||
@ -469,14 +348,14 @@ void BufferView::setCursorFromScrollbar()
|
|||||||
case bv_funcs::CUR_ABOVE:
|
case bv_funcs::CUR_ABOVE:
|
||||||
// We reset the cursor because bv_funcs::status() does not
|
// We reset the cursor because bv_funcs::status() does not
|
||||||
// work when the cursor is within mathed.
|
// work when the cursor is within mathed.
|
||||||
cur.reset(buffer_->inset());
|
cur.reset(buffer_.inset());
|
||||||
t.setCursorFromCoordinates(cur, 0, first);
|
t.setCursorFromCoordinates(cur, 0, first);
|
||||||
cur.clearSelection();
|
cur.clearSelection();
|
||||||
break;
|
break;
|
||||||
case bv_funcs::CUR_BELOW:
|
case bv_funcs::CUR_BELOW:
|
||||||
// We reset the cursor because bv_funcs::status() does not
|
// We reset the cursor because bv_funcs::status() does not
|
||||||
// work when the cursor is within mathed.
|
// work when the cursor is within mathed.
|
||||||
cur.reset(buffer_->inset());
|
cur.reset(buffer_.inset());
|
||||||
t.setCursorFromCoordinates(cur, 0, last);
|
t.setCursorFromCoordinates(cur, 0, last);
|
||||||
cur.clearSelection();
|
cur.clearSelection();
|
||||||
break;
|
break;
|
||||||
@ -484,7 +363,7 @@ void BufferView::setCursorFromScrollbar()
|
|||||||
int const y = bv_funcs::getPos(*this, cur, cur.boundary()).y_;
|
int const y = bv_funcs::getPos(*this, cur, cur.boundary()).y_;
|
||||||
int const newy = min(last, max(y, first));
|
int const newy = min(last, max(y, first));
|
||||||
if (y != newy) {
|
if (y != newy) {
|
||||||
cur.reset(buffer_->inset());
|
cur.reset(buffer_.inset());
|
||||||
t.setCursorFromCoordinates(cur, 0, newy);
|
t.setCursorFromCoordinates(cur, 0, newy);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -508,7 +387,7 @@ void BufferView::saveBookmark(unsigned int idx)
|
|||||||
// pit and pos will be updated with bottom level pit/pos
|
// pit and pos will be updated with bottom level pit/pos
|
||||||
// when lyx exits.
|
// when lyx exits.
|
||||||
LyX::ref().session().bookmarks().save(
|
LyX::ref().session().bookmarks().save(
|
||||||
FileName(buffer_->fileName()),
|
FileName(buffer_.fileName()),
|
||||||
cursor_.bottom().pit(),
|
cursor_.bottom().pit(),
|
||||||
cursor_.bottom().pos(),
|
cursor_.bottom().pos(),
|
||||||
cursor_.paragraph().id(),
|
cursor_.paragraph().id(),
|
||||||
@ -530,8 +409,8 @@ boost::tuple<pit_type, pos_type, int> BufferView::moveToPosition(pit_type bottom
|
|||||||
// This is the case for a 'live' bookmark when unique paragraph ID
|
// This is the case for a 'live' bookmark when unique paragraph ID
|
||||||
// is used to track bookmarks.
|
// is used to track bookmarks.
|
||||||
if (top_id > 0) {
|
if (top_id > 0) {
|
||||||
ParIterator par = buffer_->getParFromID(top_id);
|
ParIterator par = buffer_.getParFromID(top_id);
|
||||||
if (par != buffer_->par_iterator_end()) {
|
if (par != buffer_.par_iterator_end()) {
|
||||||
DocIterator dit = makeDocIterator(par, min(par->size(), top_pos));
|
DocIterator dit = makeDocIterator(par, min(par->size(), top_pos));
|
||||||
// Some slices of the iterator may not be
|
// Some slices of the iterator may not be
|
||||||
// reachable (e.g. closed collapsable inset)
|
// reachable (e.g. closed collapsable inset)
|
||||||
@ -556,8 +435,8 @@ boost::tuple<pit_type, pos_type, int> BufferView::moveToPosition(pit_type bottom
|
|||||||
// restoration is inaccurate. If a bookmark was within an inset,
|
// restoration is inaccurate. If a bookmark was within an inset,
|
||||||
// it will be restored to the left of the outmost inset that contains
|
// it will be restored to the left of the outmost inset that contains
|
||||||
// the bookmark.
|
// the bookmark.
|
||||||
if (static_cast<size_t>(bottom_pit) < buffer_->paragraphs().size()) {
|
if (static_cast<size_t>(bottom_pit) < buffer_.paragraphs().size()) {
|
||||||
DocIterator it = doc_iterator_begin(buffer_->inset());
|
DocIterator it = doc_iterator_begin(buffer_.inset());
|
||||||
it.pit() = bottom_pit;
|
it.pit() = bottom_pit;
|
||||||
it.pos() = min(bottom_pos, it.paragraph().size());
|
it.pos() = min(bottom_pos, it.paragraph().size());
|
||||||
setCursor(it);
|
setCursor(it);
|
||||||
@ -613,10 +492,10 @@ FuncStatus BufferView::getStatus(FuncRequest const & cmd)
|
|||||||
switch (cmd.action) {
|
switch (cmd.action) {
|
||||||
|
|
||||||
case LFUN_UNDO:
|
case LFUN_UNDO:
|
||||||
flag.enabled(!buffer_->undostack().empty());
|
flag.enabled(!buffer_.undostack().empty());
|
||||||
break;
|
break;
|
||||||
case LFUN_REDO:
|
case LFUN_REDO:
|
||||||
flag.enabled(!buffer_->redostack().empty());
|
flag.enabled(!buffer_.redostack().empty());
|
||||||
break;
|
break;
|
||||||
case LFUN_FILE_INSERT:
|
case LFUN_FILE_INSERT:
|
||||||
case LFUN_FILE_INSERT_PLAINTEXT_PARA:
|
case LFUN_FILE_INSERT_PLAINTEXT_PARA:
|
||||||
@ -656,12 +535,12 @@ FuncStatus BufferView::getStatus(FuncRequest const & cmd)
|
|||||||
|
|
||||||
case LFUN_CHANGES_TRACK:
|
case LFUN_CHANGES_TRACK:
|
||||||
flag.enabled(true);
|
flag.enabled(true);
|
||||||
flag.setOnOff(buffer_->params().trackChanges);
|
flag.setOnOff(buffer_.params().trackChanges);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_CHANGES_OUTPUT:
|
case LFUN_CHANGES_OUTPUT:
|
||||||
flag.enabled(buffer_);
|
flag.enabled(true);
|
||||||
flag.setOnOff(buffer_->params().outputChanges);
|
flag.setOnOff(buffer_.params().outputChanges);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_CHANGES_MERGE:
|
case LFUN_CHANGES_MERGE:
|
||||||
@ -672,11 +551,11 @@ FuncStatus BufferView::getStatus(FuncRequest const & cmd)
|
|||||||
// In principle, these command should only be enabled if there
|
// In principle, these command should only be enabled if there
|
||||||
// is a change in the document. However, without proper
|
// is a change in the document. However, without proper
|
||||||
// optimizations, this will inevitably result in poor performance.
|
// optimizations, this will inevitably result in poor performance.
|
||||||
flag.enabled(buffer_);
|
flag.enabled(true);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_TOGGLE_COMPRESSION: {
|
case LFUN_BUFFER_TOGGLE_COMPRESSION: {
|
||||||
flag.setOnOff(buffer_->params().compressed);
|
flag.setOnOff(buffer_.params().compressed);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -702,10 +581,6 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
<< " button[" << cmd.button() << ']'
|
<< " button[" << cmd.button() << ']'
|
||||||
<< endl;
|
<< endl;
|
||||||
|
|
||||||
// FIXME: this should not be possible.
|
|
||||||
if (!buffer_)
|
|
||||||
return Update::None;
|
|
||||||
|
|
||||||
Cursor & cur = cursor_;
|
Cursor & cur = cursor_;
|
||||||
// Default Update flags.
|
// Default Update flags.
|
||||||
Update::flags updateFlags = Update::Force | Update::FitCursor;
|
Update::flags updateFlags = Update::Force | Update::FitCursor;
|
||||||
@ -774,7 +649,9 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
case LFUN_PARAGRAPH_GOTO: {
|
case LFUN_PARAGRAPH_GOTO: {
|
||||||
int const id = convert<int>(to_utf8(cmd.argument()));
|
int const id = convert<int>(to_utf8(cmd.argument()));
|
||||||
int i = 0;
|
int i = 0;
|
||||||
for (Buffer * b = buffer_; i == 0 || b != buffer_; b = theBufferList().next(b)) {
|
for (Buffer * b = &buffer_; i == 0 || b != &buffer_;
|
||||||
|
b = theBufferList().next(b)) {
|
||||||
|
|
||||||
ParIterator par = b->getParFromID(id);
|
ParIterator par = b->getParFromID(id);
|
||||||
if (par == b->par_iterator_end()) {
|
if (par == b->par_iterator_end()) {
|
||||||
LYXERR(Debug::INFO)
|
LYXERR(Debug::INFO)
|
||||||
@ -786,7 +663,7 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
<< " found in buffer `"
|
<< " found in buffer `"
|
||||||
<< b->fileName() << "'." << endl;
|
<< b->fileName() << "'." << endl;
|
||||||
|
|
||||||
if (b == buffer_) {
|
if (b == &buffer_) {
|
||||||
// Set the cursor
|
// Set the cursor
|
||||||
setCursor(makeDocIterator(par, 0));
|
setCursor(makeDocIterator(par, 0));
|
||||||
} else {
|
} else {
|
||||||
@ -806,20 +683,20 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
case LFUN_OUTLINE_UP:
|
case LFUN_OUTLINE_UP:
|
||||||
toc::outline(toc::Up, cursor_);
|
toc::outline(toc::Up, cursor_);
|
||||||
cursor_.text()->setCursor(cursor_, cursor_.pit(), 0);
|
cursor_.text()->setCursor(cursor_, cursor_.pit(), 0);
|
||||||
updateLabels(*buffer_);
|
updateLabels(buffer_);
|
||||||
break;
|
break;
|
||||||
case LFUN_OUTLINE_DOWN:
|
case LFUN_OUTLINE_DOWN:
|
||||||
toc::outline(toc::Down, cursor_);
|
toc::outline(toc::Down, cursor_);
|
||||||
cursor_.text()->setCursor(cursor_, cursor_.pit(), 0);
|
cursor_.text()->setCursor(cursor_, cursor_.pit(), 0);
|
||||||
updateLabels(*buffer_);
|
updateLabels(buffer_);
|
||||||
break;
|
break;
|
||||||
case LFUN_OUTLINE_IN:
|
case LFUN_OUTLINE_IN:
|
||||||
toc::outline(toc::In, cursor_);
|
toc::outline(toc::In, cursor_);
|
||||||
updateLabels(*buffer_);
|
updateLabels(buffer_);
|
||||||
break;
|
break;
|
||||||
case LFUN_OUTLINE_OUT:
|
case LFUN_OUTLINE_OUT:
|
||||||
toc::outline(toc::Out, cursor_);
|
toc::outline(toc::Out, cursor_);
|
||||||
updateLabels(*buffer_);
|
updateLabels(buffer_);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_NOTE_NEXT:
|
case LFUN_NOTE_NEXT:
|
||||||
@ -835,12 +712,12 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
}
|
}
|
||||||
|
|
||||||
case LFUN_CHANGES_TRACK:
|
case LFUN_CHANGES_TRACK:
|
||||||
buffer_->params().trackChanges = !buffer_->params().trackChanges;
|
buffer_.params().trackChanges = !buffer_.params().trackChanges;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_CHANGES_OUTPUT:
|
case LFUN_CHANGES_OUTPUT:
|
||||||
buffer_->params().outputChanges = !buffer_->params().outputChanges;
|
buffer_.params().outputChanges = !buffer_.params().outputChanges;
|
||||||
if (buffer_->params().outputChanges) {
|
if (buffer_.params().outputChanges) {
|
||||||
bool dvipost = LaTeXFeatures::isAvailable("dvipost");
|
bool dvipost = LaTeXFeatures::isAvailable("dvipost");
|
||||||
bool xcolorsoul = LaTeXFeatures::isAvailable("soul") &&
|
bool xcolorsoul = LaTeXFeatures::isAvailable("soul") &&
|
||||||
LaTeXFeatures::isAvailable("xcolor");
|
LaTeXFeatures::isAvailable("xcolor");
|
||||||
@ -872,21 +749,21 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
|
|
||||||
case LFUN_ALL_CHANGES_ACCEPT:
|
case LFUN_ALL_CHANGES_ACCEPT:
|
||||||
// select complete document
|
// select complete document
|
||||||
cursor_.reset(buffer_->inset());
|
cursor_.reset(buffer_.inset());
|
||||||
cursor_.selHandle(true);
|
cursor_.selHandle(true);
|
||||||
buffer_->text().cursorBottom(cursor_);
|
buffer_.text().cursorBottom(cursor_);
|
||||||
// accept everything in a single step to support atomic undo
|
// accept everything in a single step to support atomic undo
|
||||||
buffer_->text().acceptOrRejectChanges(cursor_, Text::ACCEPT);
|
buffer_.text().acceptOrRejectChanges(cursor_, Text::ACCEPT);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_ALL_CHANGES_REJECT:
|
case LFUN_ALL_CHANGES_REJECT:
|
||||||
// select complete document
|
// select complete document
|
||||||
cursor_.reset(buffer_->inset());
|
cursor_.reset(buffer_.inset());
|
||||||
cursor_.selHandle(true);
|
cursor_.selHandle(true);
|
||||||
buffer_->text().cursorBottom(cursor_);
|
buffer_.text().cursorBottom(cursor_);
|
||||||
// reject everything in a single step to support atomic undo
|
// reject everything in a single step to support atomic undo
|
||||||
// Note: reject does not work recursively; the user may have to repeat the operation
|
// Note: reject does not work recursively; the user may have to repeat the operation
|
||||||
buffer_->text().acceptOrRejectChanges(cursor_, Text::REJECT);
|
buffer_.text().acceptOrRejectChanges(cursor_, Text::REJECT);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_WORD_FIND:
|
case LFUN_WORD_FIND:
|
||||||
@ -945,7 +822,7 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
Inset::BIBTEX_CODE);
|
Inset::BIBTEX_CODE);
|
||||||
if (inset) {
|
if (inset) {
|
||||||
if (inset->addDatabase(to_utf8(cmd.argument())))
|
if (inset->addDatabase(to_utf8(cmd.argument())))
|
||||||
buffer_->updateBibfilesCache();
|
buffer_.updateBibfilesCache();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -957,7 +834,7 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
Inset::BIBTEX_CODE);
|
Inset::BIBTEX_CODE);
|
||||||
if (inset) {
|
if (inset) {
|
||||||
if (inset->delDatabase(to_utf8(cmd.argument())))
|
if (inset->delDatabase(to_utf8(cmd.argument())))
|
||||||
buffer_->updateBibfilesCache();
|
buffer_.updateBibfilesCache();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -968,8 +845,8 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
from = cur.selectionBegin();
|
from = cur.selectionBegin();
|
||||||
to = cur.selectionEnd();
|
to = cur.selectionEnd();
|
||||||
} else {
|
} else {
|
||||||
from = doc_iterator_begin(buffer_->inset());
|
from = doc_iterator_begin(buffer_.inset());
|
||||||
to = doc_iterator_end(buffer_->inset());
|
to = doc_iterator_end(buffer_.inset());
|
||||||
}
|
}
|
||||||
int const count = countWords(from, to);
|
int const count = countWords(from, to);
|
||||||
docstring message;
|
docstring message;
|
||||||
@ -994,7 +871,7 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
|
|
||||||
case LFUN_BUFFER_TOGGLE_COMPRESSION:
|
case LFUN_BUFFER_TOGGLE_COMPRESSION:
|
||||||
// turn compression on/off
|
// turn compression on/off
|
||||||
buffer_->params().compressed = !buffer_->params().compressed;
|
buffer_.params().compressed = !buffer_.params().compressed;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_NEXT_INSET_TOGGLE: {
|
case LFUN_NEXT_INSET_TOGGLE: {
|
||||||
@ -1031,9 +908,6 @@ Update::flags BufferView::dispatch(FuncRequest const & cmd)
|
|||||||
|
|
||||||
docstring const BufferView::requestSelection()
|
docstring const BufferView::requestSelection()
|
||||||
{
|
{
|
||||||
if (!buffer_)
|
|
||||||
return docstring();
|
|
||||||
|
|
||||||
Cursor & cur = cursor_;
|
Cursor & cur = cursor_;
|
||||||
|
|
||||||
if (!cur.selection()) {
|
if (!cur.selection()) {
|
||||||
@ -1056,17 +930,15 @@ docstring const BufferView::requestSelection()
|
|||||||
|
|
||||||
void BufferView::clearSelection()
|
void BufferView::clearSelection()
|
||||||
{
|
{
|
||||||
if (buffer_) {
|
cursor_.clearSelection();
|
||||||
cursor_.clearSelection();
|
// Clear the selection buffer. Otherwise a subsequent
|
||||||
// Clear the selection buffer. Otherwise a subsequent
|
// middle-mouse-button paste would use the selection buffer,
|
||||||
// middle-mouse-button paste would use the selection buffer,
|
// not the more current external selection.
|
||||||
// not the more current external selection.
|
cap::clearSelection();
|
||||||
cap::clearSelection();
|
xsel_cache_.set = false;
|
||||||
xsel_cache_.set = false;
|
// The buffer did not really change, but this causes the
|
||||||
// The buffer did not really change, but this causes the
|
// redraw we need because we cleared the selection above.
|
||||||
// redraw we need because we cleared the selection above.
|
buffer_.changed();
|
||||||
buffer_->changed();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1078,9 +950,7 @@ void BufferView::workAreaResize(int width, int height)
|
|||||||
|
|
||||||
// The complete text metrics will be redone.
|
// The complete text metrics will be redone.
|
||||||
text_metrics_.clear();
|
text_metrics_.clear();
|
||||||
|
updateMetrics(false);
|
||||||
if (buffer_)
|
|
||||||
resize();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -1140,12 +1010,8 @@ bool BufferView::workAreaDispatch(FuncRequest const & cmd0)
|
|||||||
// LFUN_FILE_OPEN generated by drag-and-drop.
|
// LFUN_FILE_OPEN generated by drag-and-drop.
|
||||||
FuncRequest cmd = cmd0;
|
FuncRequest cmd = cmd0;
|
||||||
|
|
||||||
// E.g. Qt mouse press when no buffer
|
|
||||||
if (!buffer_)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
Cursor cur(*this);
|
Cursor cur(*this);
|
||||||
cur.push(buffer_->inset());
|
cur.push(buffer_.inset());
|
||||||
cur.selection() = cursor_.selection();
|
cur.selection() = cursor_.selection();
|
||||||
|
|
||||||
// Either the inset under the cursor or the
|
// Either the inset under the cursor or the
|
||||||
@ -1158,7 +1024,7 @@ bool BufferView::workAreaDispatch(FuncRequest const & cmd0)
|
|||||||
|
|
||||||
// Get inset under mouse, if there is one.
|
// Get inset under mouse, if there is one.
|
||||||
Inset const * covering_inset =
|
Inset const * covering_inset =
|
||||||
getCoveringInset(buffer_->text(), cmd.x, cmd.y);
|
getCoveringInset(buffer_.text(), cmd.x, cmd.y);
|
||||||
if (covering_inset == last_inset_)
|
if (covering_inset == last_inset_)
|
||||||
// Same inset, no need to do anything...
|
// Same inset, no need to do anything...
|
||||||
return false;
|
return false;
|
||||||
@ -1208,7 +1074,7 @@ bool BufferView::workAreaDispatch(FuncRequest const & cmd0)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Build temporary cursor.
|
// Build temporary cursor.
|
||||||
Inset * inset = buffer_->text().editXY(cur, cmd.x, cmd.y);
|
Inset * inset = buffer_.text().editXY(cur, cmd.x, cmd.y);
|
||||||
|
|
||||||
// Put anchor at the same position.
|
// Put anchor at the same position.
|
||||||
cur.resetAnchor();
|
cur.resetAnchor();
|
||||||
@ -1239,10 +1105,7 @@ bool BufferView::workAreaDispatch(FuncRequest const & cmd0)
|
|||||||
|
|
||||||
void BufferView::scroll(int /*lines*/)
|
void BufferView::scroll(int /*lines*/)
|
||||||
{
|
{
|
||||||
// if (!buffer_)
|
// Text const * t = buffer_.text();
|
||||||
// return;
|
|
||||||
//
|
|
||||||
// Text const * t = &buffer_->text();
|
|
||||||
// int const line_height = defaultRowHeight();
|
// int const line_height = defaultRowHeight();
|
||||||
//
|
//
|
||||||
// // The new absolute coordinate
|
// // The new absolute coordinate
|
||||||
@ -1262,21 +1125,21 @@ void BufferView::setCursorFromRow(int row)
|
|||||||
int tmpid = -1;
|
int tmpid = -1;
|
||||||
int tmppos = -1;
|
int tmppos = -1;
|
||||||
|
|
||||||
buffer_->texrow().getIdFromRow(row, tmpid, tmppos);
|
buffer_.texrow().getIdFromRow(row, tmpid, tmppos);
|
||||||
|
|
||||||
cursor_.reset(buffer_->inset());
|
cursor_.reset(buffer_.inset());
|
||||||
if (tmpid == -1)
|
if (tmpid == -1)
|
||||||
buffer_->text().setCursor(cursor_, 0, 0);
|
buffer_.text().setCursor(cursor_, 0, 0);
|
||||||
else
|
else
|
||||||
buffer_->text().setCursor(cursor_, buffer_->getParFromID(tmpid).pit(), tmppos);
|
buffer_.text().setCursor(cursor_, buffer_.getParFromID(tmpid).pit(), tmppos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void BufferView::gotoLabel(docstring const & label)
|
void BufferView::gotoLabel(docstring const & label)
|
||||||
{
|
{
|
||||||
for (InsetIterator it = inset_iterator_begin(buffer_->inset()); it; ++it) {
|
for (InsetIterator it = inset_iterator_begin(buffer_.inset()); it; ++it) {
|
||||||
vector<docstring> labels;
|
vector<docstring> labels;
|
||||||
it->getLabelList(*buffer_, labels);
|
it->getLabelList(buffer_, labels);
|
||||||
if (std::find(labels.begin(), labels.end(), label) != labels.end()) {
|
if (std::find(labels.begin(), labels.end(), label) != labels.end()) {
|
||||||
setCursor(it);
|
setCursor(it);
|
||||||
update();
|
update();
|
||||||
@ -1343,10 +1206,10 @@ bool BufferView::checkDepm(Cursor & cur, Cursor & old)
|
|||||||
if (!changed)
|
if (!changed)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
updateLabels(*buffer_);
|
updateLabels(buffer_);
|
||||||
|
|
||||||
updateMetrics(false);
|
updateMetrics(false);
|
||||||
buffer_->changed();
|
buffer_.changed();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1442,7 +1305,7 @@ ViewMetricsInfo const & BufferView::viewMetricsInfo()
|
|||||||
// FIXME: We should split-up updateMetrics() for the singlepar case.
|
// FIXME: We should split-up updateMetrics() for the singlepar case.
|
||||||
void BufferView::updateMetrics(bool singlepar)
|
void BufferView::updateMetrics(bool singlepar)
|
||||||
{
|
{
|
||||||
Text & buftext = buffer_->text();
|
Text & buftext = buffer_.text();
|
||||||
TextMetrics & tm = textMetrics(&buftext);
|
TextMetrics & tm = textMetrics(&buftext);
|
||||||
pit_type size = int(buftext.paragraphs().size());
|
pit_type size = int(buftext.paragraphs().size());
|
||||||
|
|
||||||
@ -1571,13 +1434,10 @@ void BufferView::menuInsertLyXFile(string const & filenm)
|
|||||||
// Launch a file browser
|
// Launch a file browser
|
||||||
// FIXME UNICODE
|
// FIXME UNICODE
|
||||||
string initpath = lyxrc.document_path;
|
string initpath = lyxrc.document_path;
|
||||||
|
string const trypath = buffer_.filePath();
|
||||||
if (buffer_) {
|
// If directory is writeable, use this as default.
|
||||||
string const trypath = buffer_->filePath();
|
if (isDirWriteable(FileName(trypath)))
|
||||||
// If directory is writeable, use this as default.
|
initpath = trypath;
|
||||||
if (isDirWriteable(FileName(trypath)))
|
|
||||||
initpath = trypath;
|
|
||||||
}
|
|
||||||
|
|
||||||
// FIXME UNICODE
|
// FIXME UNICODE
|
||||||
FileDialog fileDlg(_("Select LyX document to insert"),
|
FileDialog fileDlg(_("Select LyX document to insert"),
|
||||||
@ -1617,7 +1477,7 @@ void BufferView::menuInsertLyXFile(string const & filenm)
|
|||||||
docstring res;
|
docstring res;
|
||||||
Buffer buf("", false);
|
Buffer buf("", false);
|
||||||
if (lyx::loadLyXFile(&buf, FileName(filename))) {
|
if (lyx::loadLyXFile(&buf, FileName(filename))) {
|
||||||
ErrorList & el = buffer_->errorList("Parse");
|
ErrorList & el = buffer_.errorList("Parse");
|
||||||
// Copy the inserted document error list into the current buffer one.
|
// Copy the inserted document error list into the current buffer one.
|
||||||
el = buf.errorList("Parse");
|
el = buf.errorList("Parse");
|
||||||
recordUndo(cursor_);
|
recordUndo(cursor_);
|
||||||
@ -1629,8 +1489,8 @@ void BufferView::menuInsertLyXFile(string const & filenm)
|
|||||||
|
|
||||||
// emit message signal.
|
// emit message signal.
|
||||||
message(bformat(res, disp_fn));
|
message(bformat(res, disp_fn));
|
||||||
buffer_->errors("Parse");
|
buffer_.errors("Parse");
|
||||||
resize();
|
updateMetrics(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace lyx
|
} // namespace lyx
|
||||||
|
@ -79,20 +79,14 @@ struct ScrollbarParameters
|
|||||||
*/
|
*/
|
||||||
class BufferView : boost::noncopyable {
|
class BufferView : boost::noncopyable {
|
||||||
public:
|
public:
|
||||||
BufferView();
|
///
|
||||||
|
BufferView(Buffer & buffer);
|
||||||
|
|
||||||
~BufferView();
|
~BufferView();
|
||||||
|
|
||||||
/// set the buffer we are viewing.
|
|
||||||
/// \todo FIXME: eventually, we will create a new BufferView
|
|
||||||
/// when switching Buffers, so this method should go.
|
|
||||||
/// returns the buffer currently set
|
|
||||||
Buffer * setBuffer(Buffer * b);
|
|
||||||
/// return the buffer being viewed.
|
/// return the buffer being viewed.
|
||||||
Buffer * buffer() const;
|
Buffer * buffer();
|
||||||
|
Buffer const * buffer() const;
|
||||||
/// resize the BufferView.
|
|
||||||
void resize();
|
|
||||||
|
|
||||||
/// perform pending metrics updates.
|
/// perform pending metrics updates.
|
||||||
/** \c Update::FitCursor means first to do a FitCursor, and to
|
/** \c Update::FitCursor means first to do a FitCursor, and to
|
||||||
@ -269,7 +263,7 @@ private:
|
|||||||
///
|
///
|
||||||
CoordCache coord_cache_;
|
CoordCache coord_cache_;
|
||||||
///
|
///
|
||||||
Buffer * buffer_;
|
Buffer & buffer_;
|
||||||
|
|
||||||
/// Estimated average par height for scrollbar.
|
/// Estimated average par height for scrollbar.
|
||||||
int wh_;
|
int wh_;
|
||||||
|
@ -77,7 +77,15 @@ bool Importer::Import(LyXView * lv, FileName const & filename,
|
|||||||
|
|
||||||
|
|
||||||
if (loader_format == "lyx") {
|
if (loader_format == "lyx") {
|
||||||
lv->loadLyXFile(lyxfile);
|
Buffer * buf = lv->loadLyXFile(lyxfile);
|
||||||
|
if (!buf) {
|
||||||
|
// we are done
|
||||||
|
lv->message(_("file not imported!"));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
updateLabels(*buf);
|
||||||
|
lv->setBuffer(buf);
|
||||||
|
lv->showErrorList("Parse");
|
||||||
} else {
|
} else {
|
||||||
Buffer * const b = newFile(lyxfile.absFilename(), string(), true);
|
Buffer * const b = newFile(lyxfile.absFilename(), string(), true);
|
||||||
if (b)
|
if (b)
|
||||||
|
42
src/LyX.cpp
42
src/LyX.cpp
@ -435,11 +435,12 @@ int LyX::exec(int & argc, char * argv[])
|
|||||||
}
|
}
|
||||||
|
|
||||||
BufferList::iterator begin = pimpl_->buffer_list_.begin();
|
BufferList::iterator begin = pimpl_->buffer_list_.begin();
|
||||||
BufferList::iterator end = pimpl_->buffer_list_.end();
|
|
||||||
|
|
||||||
bool final_success = false;
|
bool final_success = false;
|
||||||
for (BufferList::iterator I = begin; I != end; ++I) {
|
for (BufferList::iterator I = begin; I != pimpl_->buffer_list_.end(); ++I) {
|
||||||
Buffer * buf = *I;
|
Buffer * buf = *I;
|
||||||
|
if (buf != buf->getMasterBuffer())
|
||||||
|
continue;
|
||||||
bool success = false;
|
bool success = false;
|
||||||
buf->dispatch(batch_command, &success);
|
buf->dispatch(batch_command, &success);
|
||||||
final_success |= success;
|
final_success |= success;
|
||||||
@ -634,25 +635,36 @@ void LyX::restoreGuiSession()
|
|||||||
if (!pimpl_->files_to_load_.empty()) {
|
if (!pimpl_->files_to_load_.empty()) {
|
||||||
for_each(pimpl_->files_to_load_.begin(),
|
for_each(pimpl_->files_to_load_.begin(),
|
||||||
pimpl_->files_to_load_.end(),
|
pimpl_->files_to_load_.end(),
|
||||||
bind(&LyXView::loadLyXFile, view, _1, true, false, false));
|
bind(&LyXView::loadLyXFile, view, _1, true));
|
||||||
// clear this list to save a few bytes of RAM
|
// clear this list to save a few bytes of RAM
|
||||||
pimpl_->files_to_load_.clear();
|
pimpl_->files_to_load_.clear();
|
||||||
pimpl_->session_->lastOpened().clear();
|
pimpl_->session_->lastOpened().clear();
|
||||||
return;
|
|
||||||
|
} else if (lyxrc.load_session) {
|
||||||
|
vector<FileName> const & lastopened = pimpl_->session_->lastOpened().getfiles();
|
||||||
|
// do not add to the lastfile list since these files are restored from
|
||||||
|
// last session, and should be already there (regular files), or should
|
||||||
|
// not be added at all (help files).
|
||||||
|
for_each(lastopened.begin(), lastopened.end(),
|
||||||
|
bind(&LyXView::loadLyXFile, view, _1, false));
|
||||||
|
|
||||||
|
// clear this list to save a few bytes of RAM
|
||||||
|
pimpl_->session_->lastOpened().clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!lyxrc.load_session)
|
BufferList::iterator I = pimpl_->buffer_list_.begin();
|
||||||
return;
|
BufferList::iterator end = pimpl_->buffer_list_.end();
|
||||||
|
for (; I != end; ++I) {
|
||||||
|
Buffer * buf = *I;
|
||||||
|
if (buf != buf->getMasterBuffer())
|
||||||
|
continue;
|
||||||
|
updateLabels(*buf);
|
||||||
|
}
|
||||||
|
|
||||||
vector<FileName> const & lastopened = pimpl_->session_->lastOpened().getfiles();
|
// FIXME: Switch to the last loaded Buffer. This must not be the first one
|
||||||
// do not add to the lastfile list since these files are restored from
|
// because the Buffer won't be connected in this case. The correct solution
|
||||||
// last session, and should be already there (regular files), or should
|
// would be to avoid the manual connection of the current Buffer in LyXView.
|
||||||
// not be added at all (help files).
|
view->setBuffer(pimpl_->buffer_list_.last());
|
||||||
for_each(lastopened.begin(), lastopened.end(),
|
|
||||||
bind(&LyXView::loadLyXFile, view, _1, false, false, false));
|
|
||||||
|
|
||||||
// clear this list to save a few bytes of RAM
|
|
||||||
pimpl_->session_->lastOpened().clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -124,7 +124,7 @@ void LyXAction::init()
|
|||||||
{ LFUN_BUFFER_NEW, "buffer-new", NoBuffer },
|
{ LFUN_BUFFER_NEW, "buffer-new", NoBuffer },
|
||||||
{ LFUN_BUFFER_NEW_TEMPLATE,"buffer-new-template", NoBuffer },
|
{ LFUN_BUFFER_NEW_TEMPLATE,"buffer-new-template", NoBuffer },
|
||||||
{ LFUN_BUFFER_RELOAD, "buffer-reload", ReadOnly },
|
{ LFUN_BUFFER_RELOAD, "buffer-reload", ReadOnly },
|
||||||
{ LFUN_BUFFER_SWITCH, "buffer-switch", ReadOnly },
|
{ LFUN_BUFFER_SWITCH, "buffer-switch", NoBuffer | ReadOnly },
|
||||||
{ LFUN_BUFFER_TOGGLE_READ_ONLY, "buffer-toggle-read-only", ReadOnly },
|
{ LFUN_BUFFER_TOGGLE_READ_ONLY, "buffer-toggle-read-only", ReadOnly },
|
||||||
{ LFUN_BUFFER_UPDATE, "buffer-update", ReadOnly },
|
{ LFUN_BUFFER_UPDATE, "buffer-update", ReadOnly },
|
||||||
{ LFUN_BUFFER_VIEW, "buffer-view", ReadOnly },
|
{ LFUN_BUFFER_VIEW, "buffer-view", ReadOnly },
|
||||||
|
130
src/LyXFunc.cpp
130
src/LyXFunc.cpp
@ -85,8 +85,9 @@
|
|||||||
#include "frontends/KeySymbol.h"
|
#include "frontends/KeySymbol.h"
|
||||||
#include "frontends/LyXView.h"
|
#include "frontends/LyXView.h"
|
||||||
#include "frontends/Menubar.h"
|
#include "frontends/Menubar.h"
|
||||||
#include "frontends/Toolbars.h"
|
|
||||||
#include "frontends/Selection.h"
|
#include "frontends/Selection.h"
|
||||||
|
#include "frontends/Toolbars.h"
|
||||||
|
#include "frontends/WorkArea.h"
|
||||||
|
|
||||||
#include "support/environment.h"
|
#include "support/environment.h"
|
||||||
#include "support/FileFilterList.h"
|
#include "support/FileFilterList.h"
|
||||||
@ -221,7 +222,7 @@ void LyXFunc::initKeySequences(KeyMap * kb)
|
|||||||
|
|
||||||
void LyXFunc::setLyXView(LyXView * lv)
|
void LyXFunc::setLyXView(LyXView * lv)
|
||||||
{
|
{
|
||||||
if (!quitting && lyx_view_ && lyx_view_ != lv)
|
if (!quitting && lyx_view_ && lyx_view_->view() && lyx_view_ != lv)
|
||||||
// save current selection to the selection buffer to allow
|
// save current selection to the selection buffer to allow
|
||||||
// middle-button paste in another window
|
// middle-button paste in another window
|
||||||
cap::saveSelection(lyx_view_->view()->cursor());
|
cap::saveSelection(lyx_view_->view()->cursor());
|
||||||
@ -385,6 +386,12 @@ void LyXFunc::processKeySym(KeySymbolPtr keysym, key_modifier::state state)
|
|||||||
} else {
|
} else {
|
||||||
dispatch(func);
|
dispatch(func);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* When we move around, or type, it's nice to be able to see
|
||||||
|
* the cursor immediately after the keypress.
|
||||||
|
*/
|
||||||
|
if (lyx_view_ && lyx_view_->currentWorkArea())
|
||||||
|
lyx_view_->currentWorkArea()->startBlinkingCursor();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -903,15 +910,17 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
|
|||||||
// --- Menus -----------------------------------------------
|
// --- Menus -----------------------------------------------
|
||||||
case LFUN_BUFFER_NEW:
|
case LFUN_BUFFER_NEW:
|
||||||
menuNew(argument, false);
|
menuNew(argument, false);
|
||||||
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_NEW_TEMPLATE:
|
case LFUN_BUFFER_NEW_TEMPLATE:
|
||||||
menuNew(argument, true);
|
menuNew(argument, true);
|
||||||
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_CLOSE:
|
case LFUN_BUFFER_CLOSE:
|
||||||
closeBuffer();
|
closeBuffer();
|
||||||
view()->update();
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_WRITE:
|
case LFUN_BUFFER_WRITE:
|
||||||
@ -1187,7 +1196,13 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
|
|||||||
}
|
}
|
||||||
lyx_view_->message(bformat(_("Opening help file %1$s..."),
|
lyx_view_->message(bformat(_("Opening help file %1$s..."),
|
||||||
makeDisplayPath(fname.absFilename())));
|
makeDisplayPath(fname.absFilename())));
|
||||||
lyx_view_->loadLyXFile(fname, false);
|
Buffer * buf = lyx_view_->loadLyXFile(fname, false);
|
||||||
|
if (buf) {
|
||||||
|
updateLabels(*buf);
|
||||||
|
lyx_view_->setBuffer(buf);
|
||||||
|
lyx_view_->showErrorList("Parse");
|
||||||
|
}
|
||||||
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1241,29 +1256,31 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
|
|||||||
case LFUN_BUFFER_SWITCH:
|
case LFUN_BUFFER_SWITCH:
|
||||||
BOOST_ASSERT(lyx_view_);
|
BOOST_ASSERT(lyx_view_);
|
||||||
lyx_view_->setBuffer(theBufferList().getBuffer(argument));
|
lyx_view_->setBuffer(theBufferList().getBuffer(argument));
|
||||||
updateFlags = Update::Force;
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_NEXT:
|
case LFUN_BUFFER_NEXT:
|
||||||
BOOST_ASSERT(lyx_view_);
|
BOOST_ASSERT(lyx_view_);
|
||||||
lyx_view_->setBuffer(theBufferList().next(lyx_view_->buffer()));
|
lyx_view_->setBuffer(theBufferList().next(lyx_view_->buffer()));
|
||||||
updateFlags = Update::Force;
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_BUFFER_PREVIOUS:
|
case LFUN_BUFFER_PREVIOUS:
|
||||||
BOOST_ASSERT(lyx_view_);
|
BOOST_ASSERT(lyx_view_);
|
||||||
lyx_view_->setBuffer(theBufferList().previous(lyx_view_->buffer()));
|
lyx_view_->setBuffer(theBufferList().previous(lyx_view_->buffer()));
|
||||||
updateFlags = Update::Force;
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_FILE_NEW:
|
case LFUN_FILE_NEW:
|
||||||
BOOST_ASSERT(lyx_view_);
|
BOOST_ASSERT(lyx_view_);
|
||||||
newFile(*lyx_view_, argument);
|
newFile(*lyx_view_, argument);
|
||||||
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_FILE_OPEN:
|
case LFUN_FILE_OPEN:
|
||||||
BOOST_ASSERT(lyx_view_);
|
BOOST_ASSERT(lyx_view_);
|
||||||
open(argument);
|
open(argument);
|
||||||
|
updateFlags = Update::None;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case LFUN_DROP_LAYOUTS_CHOICE:
|
case LFUN_DROP_LAYOUTS_CHOICE:
|
||||||
@ -1296,24 +1313,35 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
|
|||||||
int row;
|
int row;
|
||||||
istringstream is(argument);
|
istringstream is(argument);
|
||||||
is >> file_name >> row;
|
is >> file_name >> row;
|
||||||
if (prefixIs(file_name, package().temp_dir().absFilename())) {
|
Buffer * buf = 0;
|
||||||
|
bool loaded = false;
|
||||||
|
if (prefixIs(file_name, package().temp_dir().absFilename()))
|
||||||
// Needed by inverse dvi search. If it is a file
|
// Needed by inverse dvi search. If it is a file
|
||||||
// in tmpdir, call the apropriated function
|
// in tmpdir, call the apropriated function
|
||||||
lyx_view_->setBuffer(theBufferList().getBufferFromTmp(file_name));
|
buf = theBufferList().getBufferFromTmp(file_name);
|
||||||
} else {
|
else {
|
||||||
// Must replace extension of the file to be .lyx
|
// Must replace extension of the file to be .lyx
|
||||||
// and get full path
|
// and get full path
|
||||||
FileName const s = fileSearch(string(), changeExtension(file_name, ".lyx"), "lyx");
|
FileName const s = fileSearch(string(), changeExtension(file_name, ".lyx"), "lyx");
|
||||||
// Either change buffer or load the file
|
// Either change buffer or load the file
|
||||||
if (theBufferList().exists(s.absFilename())) {
|
if (theBufferList().exists(s.absFilename()))
|
||||||
lyx_view_->setBuffer(theBufferList().getBuffer(s.absFilename()));
|
buf = theBufferList().getBuffer(s.absFilename());
|
||||||
} else {
|
else {
|
||||||
lyx_view_->loadLyXFile(s);
|
buf = lyx_view_->loadLyXFile(s);
|
||||||
|
loaded = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
view()->setCursorFromRow(row);
|
if (!buf) {
|
||||||
|
updateFlags = Update::None;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
updateLabels(*buf);
|
||||||
|
lyx_view_->setBuffer(buf);
|
||||||
|
view()->setCursorFromRow(row);
|
||||||
|
if (loaded)
|
||||||
|
lyx_view_->showErrorList("Parse");
|
||||||
updateFlags = Update::FitCursor;
|
updateFlags = Update::FitCursor;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1472,36 +1500,33 @@ void LyXFunc::dispatch(FuncRequest const & cmd)
|
|||||||
}
|
}
|
||||||
|
|
||||||
case LFUN_BUFFER_CHILD_OPEN: {
|
case LFUN_BUFFER_CHILD_OPEN: {
|
||||||
// takes an optional argument, "|bool", at the end
|
BOOST_ASSERT(lyx_view_ && lyx_view_->buffer());
|
||||||
// indicating whether this file is being opened automatically
|
Buffer * parent = lyx_view_->buffer();
|
||||||
// by LyX itself, in which case we will not want to switch
|
FileName filename = makeAbsPath(argument, parent->filePath());
|
||||||
// buffers after opening. The default is false, so in practice
|
|
||||||
// it is used only when true.
|
|
||||||
BOOST_ASSERT(lyx_view_);
|
|
||||||
int const arglength = argument.length();
|
|
||||||
FileName filename;
|
|
||||||
bool autoOpen = false;
|
|
||||||
if (argument.substr(arglength - 5, 5) == "|true") {
|
|
||||||
autoOpen = true;
|
|
||||||
filename = makeAbsPath(argument.substr(0, arglength - 5),
|
|
||||||
lyx_view_->buffer()->filePath());
|
|
||||||
} else if (argument.substr(arglength - 6, 6) == "|false") {
|
|
||||||
filename = makeAbsPath(argument.substr(0, arglength - 6),
|
|
||||||
lyx_view_->buffer()->filePath());
|
|
||||||
} else filename =
|
|
||||||
makeAbsPath(argument, lyx_view_->buffer()->filePath());
|
|
||||||
view()->saveBookmark(false);
|
view()->saveBookmark(false);
|
||||||
|
Buffer * child = 0;
|
||||||
|
bool parsed = false;
|
||||||
if (theBufferList().exists(filename.absFilename())) {
|
if (theBufferList().exists(filename.absFilename())) {
|
||||||
Buffer * buf = theBufferList().getBuffer(filename.absFilename());
|
child = theBufferList().getBuffer(filename.absFilename());
|
||||||
if (!autoOpen)
|
} else {
|
||||||
lyx_view_->setBuffer(buf, true);
|
setMessage(bformat(_("Opening child document %1$s..."),
|
||||||
else
|
makeDisplayPath(filename.absFilename())));
|
||||||
buf->setParentName(lyx_view_->buffer()->fileName());
|
child = lyx_view_->loadLyXFile(filename, true);
|
||||||
} else
|
parsed = true;
|
||||||
lyx_view_->loadLyXFile(filename, true, true, autoOpen);
|
}
|
||||||
|
if (child) {
|
||||||
|
// Set the parent name of the child document.
|
||||||
|
// This makes insertion of citations and references in the child work,
|
||||||
|
// when the target is in the parent or another child document.
|
||||||
|
child->setParentName(parent->fileName());
|
||||||
|
updateLabels(*child->getMasterBuffer());
|
||||||
|
lyx_view_->setBuffer(child);
|
||||||
|
if (parsed)
|
||||||
|
lyx_view_->showErrorList("Parse");
|
||||||
|
}
|
||||||
|
|
||||||
// If a screen update is required (in case where auto_open is false),
|
// If a screen update is required (in case where auto_open is false),
|
||||||
// loadLyXFile() would have taken care of it already. Otherwise we shall
|
// setBuffer() would have taken care of it already. Otherwise we shall
|
||||||
// reset the update flag because it can cause a circular problem.
|
// reset the update flag because it can cause a circular problem.
|
||||||
// See bug 3970.
|
// See bug 3970.
|
||||||
updateFlags = Update::None;
|
updateFlags = Update::None;
|
||||||
@ -1997,10 +2022,8 @@ void LyXFunc::menuNew(string const & name, bool fromTemplate)
|
|||||||
}
|
}
|
||||||
|
|
||||||
Buffer * const b = newFile(filename, templname, !name.empty());
|
Buffer * const b = newFile(filename, templname, !name.empty());
|
||||||
if (b) {
|
if (b)
|
||||||
updateLabels(*b);
|
|
||||||
lyx_view_->setBuffer(b);
|
lyx_view_->setBuffer(b);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -2060,7 +2083,11 @@ void LyXFunc::open(string const & fname)
|
|||||||
lyx_view_->message(bformat(_("Opening document %1$s..."), disp_fn));
|
lyx_view_->message(bformat(_("Opening document %1$s..."), disp_fn));
|
||||||
|
|
||||||
docstring str2;
|
docstring str2;
|
||||||
if (lyx_view_->loadLyXFile(fullname)) {
|
Buffer * buf = lyx_view_->loadLyXFile(fullname);
|
||||||
|
if (buf) {
|
||||||
|
updateLabels(*buf);
|
||||||
|
lyx_view_->setBuffer(buf);
|
||||||
|
lyx_view_->showErrorList("Parse");
|
||||||
str2 = bformat(_("Document %1$s opened."), disp_fn);
|
str2 = bformat(_("Document %1$s opened."), disp_fn);
|
||||||
} else {
|
} else {
|
||||||
str2 = bformat(_("Could not open document %1$s"), disp_fn);
|
str2 = bformat(_("Could not open document %1$s"), disp_fn);
|
||||||
@ -2169,8 +2196,19 @@ void LyXFunc::closeBuffer()
|
|||||||
void LyXFunc::reloadBuffer()
|
void LyXFunc::reloadBuffer()
|
||||||
{
|
{
|
||||||
FileName filename(lyx_view_->buffer()->fileName());
|
FileName filename(lyx_view_->buffer()->fileName());
|
||||||
|
docstring const disp_fn = makeDisplayPath(filename.absFilename());
|
||||||
|
docstring str;
|
||||||
closeBuffer();
|
closeBuffer();
|
||||||
lyx_view_->loadLyXFile(filename);
|
Buffer * buf = lyx_view_->loadLyXFile(filename);
|
||||||
|
if (buf) {
|
||||||
|
updateLabels(*buf);
|
||||||
|
lyx_view_->setBuffer(buf);
|
||||||
|
lyx_view_->showErrorList("Parse");
|
||||||
|
str = bformat(_("Document %1$s reloaded."), disp_fn);
|
||||||
|
} else {
|
||||||
|
str = bformat(_("Could not reload document %1$s"), disp_fn);
|
||||||
|
}
|
||||||
|
lyx_view_->message(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Each "lyx_view_" should have it's own message method. lyxview and
|
// Each "lyx_view_" should have it's own message method. lyxview and
|
||||||
|
@ -52,14 +52,11 @@ LyXView & Application::createView(unsigned int width,
|
|||||||
const std::string & geometryArg)
|
const std::string & geometryArg)
|
||||||
{
|
{
|
||||||
LyXView & view = gui().createRegisteredView();
|
LyXView & view = gui().createRegisteredView();
|
||||||
int view_id = view.id();
|
|
||||||
|
|
||||||
theLyXFunc().setLyXView(&view);
|
theLyXFunc().setLyXView(&view);
|
||||||
|
|
||||||
/*int workArea_id_ =*/ gui().newWorkArea(width, height, view_id);
|
|
||||||
|
|
||||||
view.init();
|
view.init();
|
||||||
view.setGeometry(width, height, posx, posy, maximized, iconSizeXY, geometryArg);
|
view.setGeometry(width, height, posx, posy, maximized, iconSizeXY, geometryArg);
|
||||||
|
|
||||||
view.setFocus();
|
view.setFocus();
|
||||||
|
|
||||||
setCurrentView(view);
|
setCurrentView(view);
|
||||||
|
@ -34,29 +34,56 @@ class Selection;
|
|||||||
There should be only one instance of this class. No Qt object
|
There should be only one instance of this class. No Qt object
|
||||||
initialisation should be done before the instanciation of this class.
|
initialisation should be done before the instanciation of this class.
|
||||||
|
|
||||||
\todo The work areas handling could be moved to a base virtual class
|
Model/View/Controller separation at frontend level in LyX-qt4:
|
||||||
common to all frontends.
|
|
||||||
|
BufferList (N Buffers)
|
||||||
|
|
|
||||||
|
Buffer-a
|
||||||
|
Buffer-b
|
||||||
|
Buffer-c
|
||||||
|
Buffer-d
|
||||||
|
|
||||||
|
Application (this is the frontend really, should probably be renamed).
|
||||||
|
|
|
||||||
|
LyXView-1 (M1 WorkAreas, M1 <= N)
|
||||||
|
| |
|
||||||
|
| <tab-widget>
|
||||||
|
| | (many)
|
||||||
|
| WorkArea-1
|
||||||
|
| |
|
||||||
|
| BufferView <-----------> Buffer-c
|
||||||
|
| |
|
||||||
|
| Cursor
|
||||||
|
|
|
||||||
|
LyXView-2 (M2 WorkAreas, M2 <= N, M2 independent of M1)
|
||||||
|
|
|
||||||
|
...
|
||||||
|
|
||||||
Model/View/Controller separation in LyX:
|
|
||||||
|
|
||||||
1) The Model: \c Buffer
|
1) The Model: \c Buffer
|
||||||
|
|
||||||
The Buffer is the in-memory representation of a LyX file format. The
|
The Buffer is the in-memory representation of a LyX file format. The
|
||||||
Buffer does not (should not) have any information on what part of it
|
Buffer does not (should not) have any information on what part of it
|
||||||
is represented on screen. There is one unique Buffer per opened LyX
|
is represented on screen. There is one unique Buffer per opened LyX
|
||||||
file.
|
file. A Buffer may or may not be represented on screen; typically, a
|
||||||
|
child document does not have an associated BufferView unless the user
|
||||||
|
choose to visualize it.
|
||||||
|
|
||||||
|
|
||||||
2) The Controller: \c BufferView / \c Painter
|
2) The Controller: \c BufferView / \c Painter \c Cursor
|
||||||
|
|
||||||
The BufferView is a tool used by the view that translates a part of
|
The BufferView is a tool used by the view (\sa WorkArea) that
|
||||||
the Buffer contents into drawing routines. The BufferView asks each
|
translates a part of the Buffer contents into drawing routines. The
|
||||||
inset of the Buffer to draw itself onto the screen using the Painter.
|
BufferView asks each inset of the Buffer to draw itself onto the
|
||||||
There can be only one Buffer displayed in a BufferView. While there
|
screen using the Painter. There can be only one Buffer displayed in
|
||||||
is the possibility to switch Buffer inside the BufferView, the goal
|
a BufferView and it is set on construction. Ideally, a BufferView
|
||||||
is to instantiate a new BufferView on each Buffer switch.
|
should not be able to change the contents of its associated Buffer.
|
||||||
|
A BufferView is instanciated and destroyed by a \c WorkArea; it is
|
||||||
|
automatically destroyed by the parent WorkArea when its Buffer is
|
||||||
|
closed.
|
||||||
|
|
||||||
\todo Instantiate a new BufferView on each Buffer switch.
|
\todo Move all Buffer changing LFUN to LyXFunc or Cursor.
|
||||||
|
\todo BufferView::buffer() should only offer const access.
|
||||||
|
|
||||||
The \c Painter is just a virtual interface to formalize each kind of
|
The \c Painter is just a virtual interface to formalize each kind of
|
||||||
drawing routines (text, line, rectangle, etc).
|
drawing routines (text, line, rectangle, etc).
|
||||||
@ -69,9 +96,10 @@ common to all frontends.
|
|||||||
3) The View: \c WorkArea (and it's qt4 specialisation GuiWorkArea)
|
3) The View: \c WorkArea (and it's qt4 specialisation GuiWorkArea)
|
||||||
|
|
||||||
This contains the real screen area where the drawing is done by the
|
This contains the real screen area where the drawing is done by the
|
||||||
Painter. One WorkArea holds one unique \c BufferView. While it could be
|
Painter. One WorkArea holds one unique \c BufferView. While it could
|
||||||
possible that multiple WorkArea share one BufferView, this is not
|
be possible that multiple WorkArea share one BufferView, this is not
|
||||||
possible right now.
|
something desirable because a BufferView is dependent of the WorkArea
|
||||||
|
size.
|
||||||
The WorkArea also provide a scrollbar which position is translated
|
The WorkArea also provide a scrollbar which position is translated
|
||||||
into scrolling command to the inner \c BufferView.
|
into scrolling command to the inner \c BufferView.
|
||||||
|
|
||||||
@ -84,18 +112,32 @@ common to all frontends.
|
|||||||
|
|
||||||
4) The Window: \c LyXView (and its qt4 specialisation \c GuiView)
|
4) The Window: \c LyXView (and its qt4 specialisation \c GuiView)
|
||||||
|
|
||||||
This is a full window containing a menubar, toolbars, a tabbar and a
|
This is a full window containing a menubar, toolbars and a central
|
||||||
WorkArea. One LyXView could in theory contain multiple WorkArea
|
widget. A LyXView is in charge of creating and closing a View for a
|
||||||
(ex: with split window) but this number is limited to one only for
|
given Buffer.
|
||||||
now. In any case, there would be only one WorkArea that gets the focus
|
In the qt4 specialisation, \c GuiView, the central widget is a tab
|
||||||
|
widget. Each tab is reverved to the visualisation of one Buffer and
|
||||||
|
contains one WorkArea. In the qt4 frontend, one LyXView thus contains
|
||||||
|
multiple WorkAreas but this number can limited to one for another
|
||||||
|
frontend. The idea is that the kernel should not know how a Buffer
|
||||||
|
is displayed on screen; it's the frontend business.
|
||||||
|
In the future, we may also have multiple Workareas showing
|
||||||
|
simultaneously in the same GuiView (ex: with split window).
|
||||||
|
|
||||||
|
\todo Implement split-window
|
||||||
|
|
||||||
|
In any case, there would be only one WorkArea that gets the focus
|
||||||
at a time.
|
at a time.
|
||||||
|
|
||||||
Now, concerning the TabBar versus TabWidget issue. Right now, there is
|
With our current implementation using a QTabWidget, each Tab own its
|
||||||
only one WorkArea and the TabBar just used to tell the BufferView inside
|
own \c WorkArea. Clicking on a tab switch a WorkArea and not really
|
||||||
the WorkArea to switch to this another Buffer.
|
a Buffer. LFUN_BUFFER_SWITCH will tell the frontend to search the
|
||||||
|
WorkArea associated to this Buffer. The WorkArea is automatically
|
||||||
|
created if not already present.
|
||||||
|
|
||||||
|
A WorkArea is connected to the Buffer::closing signal and is thus
|
||||||
|
automatically destroyed when its Buffer is closed.
|
||||||
|
|
||||||
With a TabWidget, each Tab would own its own \c WorkArea. Clicking on a tab
|
|
||||||
would switch a WorkArea instead of a Buffer.
|
|
||||||
*/
|
*/
|
||||||
class Application
|
class Application
|
||||||
{
|
{
|
||||||
|
@ -52,11 +52,6 @@ public:
|
|||||||
return view_ids_;
|
return view_ids_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
virtual int newWorkArea(unsigned int width, unsigned int height, int view_id) = 0;
|
|
||||||
///
|
|
||||||
virtual WorkArea & workArea(int id) = 0;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
std::vector<int> view_ids_;
|
std::vector<int> view_ids_;
|
||||||
|
@ -70,8 +70,7 @@ namespace frontend {
|
|||||||
docstring current_layout;
|
docstring current_layout;
|
||||||
|
|
||||||
LyXView::LyXView(int id)
|
LyXView::LyXView(int id)
|
||||||
: work_area_(0),
|
: toolbars_(new Toolbars(*this)),
|
||||||
toolbars_(new Toolbars(*this)),
|
|
||||||
autosave_timeout_(new Timeout(5000)),
|
autosave_timeout_(new Timeout(5000)),
|
||||||
dialogs_(new Dialogs(*this)),
|
dialogs_(new Dialogs(*this)),
|
||||||
controlcommand_(new ControlCommandBuffer(*this)), id_(id)
|
controlcommand_(new ControlCommandBuffer(*this)), id_(id)
|
||||||
@ -88,177 +87,95 @@ LyXView::LyXView(int id)
|
|||||||
LyXView::~LyXView()
|
LyXView::~LyXView()
|
||||||
{
|
{
|
||||||
disconnectBuffer();
|
disconnectBuffer();
|
||||||
|
disconnectBufferView();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// FIXME, there's only one WorkArea per LyXView possible for now.
|
Buffer * LyXView::buffer()
|
||||||
void LyXView::setWorkArea(WorkArea * work_area)
|
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(work_area);
|
WorkArea * work_area = currentWorkArea();
|
||||||
work_area_ = work_area;
|
if (work_area)
|
||||||
work_area_ids_.clear();
|
return work_area->bufferView().buffer();
|
||||||
work_area_ids_.push_back(work_area_->id());
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// FIXME, there's only one WorkArea per LyXView possible for now.
|
Buffer const * LyXView::buffer() const
|
||||||
WorkArea const * LyXView::currentWorkArea() const
|
|
||||||
{
|
{
|
||||||
return work_area_;
|
WorkArea const * work_area = currentWorkArea();
|
||||||
|
if (work_area)
|
||||||
|
return work_area->bufferView().buffer();
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// FIXME, there's only one WorkArea per LyXView possible for now.
|
void LyXView::setBuffer(Buffer * newBuffer)
|
||||||
WorkArea * LyXView::currentWorkArea()
|
|
||||||
{
|
|
||||||
return work_area_;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Buffer * LyXView::buffer() const
|
|
||||||
{
|
|
||||||
BOOST_ASSERT(work_area_);
|
|
||||||
return work_area_->bufferView().buffer();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void LyXView::setBuffer(Buffer * b, bool child_document)
|
|
||||||
{
|
{
|
||||||
busy(true);
|
busy(true);
|
||||||
|
|
||||||
BOOST_ASSERT(work_area_);
|
Buffer * oldBuffer = buffer();
|
||||||
Buffer * oldBuffer = work_area_->bufferView().buffer();
|
if (oldBuffer == newBuffer) {
|
||||||
|
busy(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// parentfilename will be used in case when we switch to a child
|
// parentfilename will be used in case when we switch to a child
|
||||||
// document (hence when child_document is true)
|
// document (hence when child_document is true)
|
||||||
string parentfilename;
|
string parentfilename;
|
||||||
if (oldBuffer)
|
if (oldBuffer)
|
||||||
parentfilename = oldBuffer->fileName();
|
parentfilename = oldBuffer->fileName();
|
||||||
|
|
||||||
if (!b && theBufferList().empty())
|
WorkArea * wa = workArea(*newBuffer);
|
||||||
getDialogs().hideBufferDependent();
|
if (wa == 0) {
|
||||||
|
updateLabels(*newBuffer->getMasterBuffer());
|
||||||
Buffer * newBuffer = work_area_->bufferView().setBuffer(b);
|
wa = addWorkArea(*newBuffer);
|
||||||
|
|
||||||
if (newBuffer) {
|
|
||||||
//Are we closing an oldBuffer which was a child document?
|
|
||||||
if (!b && oldBuffer && oldBuffer->getMasterBuffer() != oldBuffer)
|
|
||||||
// Update the labels and section numbering of its master Buffer.
|
|
||||||
updateLabels(*oldBuffer->getMasterBuffer());
|
|
||||||
//Are we opening a new child document?
|
|
||||||
else if (child_document && newBuffer->getMasterBuffer() != oldBuffer) {
|
|
||||||
// Set the parent name of the child document.
|
|
||||||
// This makes insertion of citations and references in the child work,
|
|
||||||
// when the target is in the parent or another child document.
|
|
||||||
newBuffer->setParentName(parentfilename);
|
|
||||||
// Update the labels and section numbering to the new master Buffer.
|
|
||||||
updateLabels(*newBuffer->getMasterBuffer());
|
|
||||||
}
|
|
||||||
//Now that all the updating of the old buffer has been done, we can
|
|
||||||
//connect the new buffer. Note that this will also disconnect the old
|
|
||||||
//buffer, if such there is.
|
|
||||||
//FIXME Is it clear that this should go right here? Or should it go
|
|
||||||
//earlier before the previous if (in which case we'd remove the "else")?
|
|
||||||
connectBuffer(*newBuffer);
|
|
||||||
|
|
||||||
/* FIXME: We need to rebuild the Toc dialog before the others even
|
|
||||||
if it will be rebuilt again in the next line. This avoid a crash when
|
|
||||||
other dialogs are rebuilt before the Toc dialog. The reason is
|
|
||||||
that closing a Buffer triggers an update of all opened dialogs
|
|
||||||
when dispatching LFUN_DIALOG_UPDATE (hence the patch).
|
|
||||||
The path is as following:
|
|
||||||
setBuffer() -> updateBufferDependent() -> RestoreButton() -> LFUN
|
|
||||||
The problem here is that the Toc dialog has not been
|
|
||||||
reconstructed (because it comes after in the list of dialogs). */
|
|
||||||
updateToc();
|
|
||||||
|
|
||||||
// Buffer-dependent dialogs should be updated or
|
|
||||||
// hidden. This should go here because some dialogs (eg ToC)
|
|
||||||
// require bv_->text.
|
|
||||||
getDialogs().updateBufferDependent(true);
|
|
||||||
} else
|
} else
|
||||||
//Disconnect the old buffer...there's no new one.
|
//Disconnect the old buffer...there's no new one.
|
||||||
disconnectBuffer();
|
disconnectBuffer();
|
||||||
|
connectBuffer(*newBuffer);
|
||||||
|
connectBufferView(wa->bufferView());
|
||||||
|
setCurrentWorkArea(wa);
|
||||||
|
|
||||||
if (quitting)
|
|
||||||
return;
|
|
||||||
|
|
||||||
updateMenubar();
|
|
||||||
updateToolbars();
|
|
||||||
updateLayoutChoice();
|
|
||||||
updateWindowTitle();
|
|
||||||
updateStatusBar();
|
|
||||||
updateTab();
|
|
||||||
busy(false);
|
busy(false);
|
||||||
work_area_->redraw();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool LyXView::loadLyXFile(FileName const & filename, bool tolastfiles,
|
Buffer * LyXView::loadLyXFile(FileName const & filename, bool tolastfiles)
|
||||||
bool child_document, bool auto_open)
|
|
||||||
{
|
{
|
||||||
busy(true);
|
busy(true);
|
||||||
|
|
||||||
BOOST_ASSERT(work_area_);
|
|
||||||
string parentfilename;
|
string parentfilename;
|
||||||
Buffer * oldBuffer = work_area_->bufferView().buffer();
|
Buffer * oldBuffer = buffer();
|
||||||
if (oldBuffer)
|
if (oldBuffer)
|
||||||
parentfilename = oldBuffer->fileName();
|
parentfilename = oldBuffer->fileName();
|
||||||
|
|
||||||
bool alreadyLoaded = checkIfLoaded(filename);
|
|
||||||
Buffer * newBuffer = checkAndLoadLyXFile(filename);
|
Buffer * newBuffer = checkAndLoadLyXFile(filename);
|
||||||
|
|
||||||
if (!newBuffer) {
|
if (!newBuffer) {
|
||||||
message(_("Document not loaded."));
|
message(_("Document not loaded."));
|
||||||
updateStatusBar();
|
updateStatusBar();
|
||||||
busy(false);
|
busy(false);
|
||||||
work_area_->redraw();
|
return 0;
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (child_document && newBuffer != oldBuffer) {
|
WorkArea * wa = addWorkArea(*newBuffer);
|
||||||
// Set the parent name of the child document.
|
|
||||||
// This makes insertion of citations and references in the child work,
|
|
||||||
// when the target is in the parent or another child document.
|
|
||||||
newBuffer->setParentName(parentfilename);
|
|
||||||
message(bformat(_("Opening child document %1$s..."),
|
|
||||||
makeDisplayPath(filename.absFilename())));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the labels and section numbering.
|
// scroll to the position when the file was last closed
|
||||||
updateLabels(*newBuffer->getMasterBuffer());
|
if (lyxrc.use_lastfilepos) {
|
||||||
|
pit_type pit;
|
||||||
bool const parse_error = !newBuffer->errorList("Parse").empty();
|
pos_type pos;
|
||||||
bool const need_switch = parse_error || !auto_open;
|
boost::tie(pit, pos) = LyX::ref().session().lastFilePos().load(filename);
|
||||||
if (need_switch) {
|
// if successfully move to pit (returned par_id is not zero),
|
||||||
setBuffer(newBuffer, child_document);
|
// update metrics and reset font
|
||||||
if (!alreadyLoaded) {
|
BufferView & bv = wa->bufferView();
|
||||||
if (parse_error)
|
if (bv.moveToPosition(pit, pos, 0, 0).get<1>()) {
|
||||||
showErrorList("Parse");
|
if (bv.fitCursor())
|
||||||
// scroll to the position when the file was last closed
|
bv.updateMetrics(false);
|
||||||
if (lyxrc.use_lastfilepos) {
|
newBuffer->text().setCurrentFont(bv.cursor());
|
||||||
pit_type pit;
|
|
||||||
pos_type pos;
|
|
||||||
boost::tie(pit, pos) = LyX::ref().session().lastFilePos().load(filename);
|
|
||||||
// if successfully move to pit (returned par_id is not zero),
|
|
||||||
// update metrics and reset font
|
|
||||||
if (work_area_->bufferView().moveToPosition(pit, pos, 0, 0).get<1>()) {
|
|
||||||
if (work_area_->bufferView().fitCursor())
|
|
||||||
work_area_->bufferView().updateMetrics(false);
|
|
||||||
newBuffer->text().setCurrentFont(work_area_->bufferView().cursor());
|
|
||||||
updateMenubar();
|
|
||||||
updateToolbars();
|
|
||||||
updateLayoutChoice();
|
|
||||||
updateStatusBar();
|
|
||||||
work_area_->redraw();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (tolastfiles)
|
|
||||||
LyX::ref().session().lastFiles().add(filename);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
busy(false);
|
busy(false);
|
||||||
return true;
|
return newBuffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -267,11 +184,6 @@ void LyXView::connectBuffer(Buffer & buf)
|
|||||||
if (errorsConnection_.connected())
|
if (errorsConnection_.connected())
|
||||||
disconnectBuffer();
|
disconnectBuffer();
|
||||||
|
|
||||||
BOOST_ASSERT(work_area_);
|
|
||||||
bufferChangedConnection_ =
|
|
||||||
buf.changed.connect(
|
|
||||||
boost::bind(&WorkArea::redraw, work_area_));
|
|
||||||
|
|
||||||
bufferStructureChangedConnection_ =
|
bufferStructureChangedConnection_ =
|
||||||
buf.getMasterBuffer()->structureChanged.connect(
|
buf.getMasterBuffer()->structureChanged.connect(
|
||||||
boost::bind(&LyXView::updateToc, this));
|
boost::bind(&LyXView::updateToc, this));
|
||||||
@ -299,30 +211,26 @@ void LyXView::connectBuffer(Buffer & buf)
|
|||||||
readonlyConnection_ =
|
readonlyConnection_ =
|
||||||
buf.readonly.connect(
|
buf.readonly.connect(
|
||||||
boost::bind(&LyXView::showReadonly, this, _1));
|
boost::bind(&LyXView::showReadonly, this, _1));
|
||||||
|
|
||||||
closingConnection_ =
|
|
||||||
buf.closing.connect(
|
|
||||||
boost::bind(&LyXView::setBuffer, this, (Buffer *)0, false));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void LyXView::disconnectBuffer()
|
void LyXView::disconnectBuffer()
|
||||||
{
|
{
|
||||||
errorsConnection_.disconnect();
|
errorsConnection_.disconnect();
|
||||||
bufferChangedConnection_.disconnect();
|
|
||||||
bufferStructureChangedConnection_.disconnect();
|
bufferStructureChangedConnection_.disconnect();
|
||||||
messageConnection_.disconnect();
|
messageConnection_.disconnect();
|
||||||
busyConnection_.disconnect();
|
busyConnection_.disconnect();
|
||||||
titleConnection_.disconnect();
|
titleConnection_.disconnect();
|
||||||
timerConnection_.disconnect();
|
timerConnection_.disconnect();
|
||||||
readonlyConnection_.disconnect();
|
readonlyConnection_.disconnect();
|
||||||
closingConnection_.disconnect();
|
|
||||||
layout_changed_connection_.disconnect();
|
layout_changed_connection_.disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void LyXView::connectBufferView(BufferView & bv)
|
void LyXView::connectBufferView(BufferView & bv)
|
||||||
{
|
{
|
||||||
|
message_connection_ = bv.message.connect(
|
||||||
|
boost::bind(&LyXView::message, this, _1));
|
||||||
show_dialog_connection_ = bv.showDialog.connect(
|
show_dialog_connection_ = bv.showDialog.connect(
|
||||||
boost::bind(&LyXView::showDialog, this, _1));
|
boost::bind(&LyXView::showDialog, this, _1));
|
||||||
show_dialog_with_data_connection_ = bv.showDialogWithData.connect(
|
show_dialog_with_data_connection_ = bv.showDialogWithData.connect(
|
||||||
@ -338,6 +246,7 @@ void LyXView::connectBufferView(BufferView & bv)
|
|||||||
|
|
||||||
void LyXView::disconnectBufferView()
|
void LyXView::disconnectBufferView()
|
||||||
{
|
{
|
||||||
|
message_connection_.disconnect();
|
||||||
show_dialog_connection_.disconnect();
|
show_dialog_connection_.disconnect();
|
||||||
show_dialog_with_data_connection_.disconnect();
|
show_dialog_with_data_connection_.disconnect();
|
||||||
show_inset_dialog_connection_.disconnect();
|
show_inset_dialog_connection_.disconnect();
|
||||||
@ -387,10 +296,10 @@ void LyXView::showReadonly(bool)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
BufferView * LyXView::view() const
|
BufferView * LyXView::view()
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(work_area_);
|
WorkArea * wa = currentWorkArea();
|
||||||
return &work_area_->bufferView();
|
return wa? &wa->bufferView() : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -402,16 +311,20 @@ void LyXView::updateToc()
|
|||||||
|
|
||||||
void LyXView::updateToolbars()
|
void LyXView::updateToolbars()
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(work_area_);
|
WorkArea * wa = currentWorkArea();
|
||||||
bool const math =
|
if (wa) {
|
||||||
work_area_->bufferView().cursor().inMathed();
|
bool const math =
|
||||||
bool const table =
|
wa->bufferView().cursor().inMathed();
|
||||||
lyx::getStatus(FuncRequest(LFUN_LAYOUT_TABULAR)).enabled();
|
bool const table =
|
||||||
bool const review =
|
lyx::getStatus(FuncRequest(LFUN_LAYOUT_TABULAR)).enabled();
|
||||||
lyx::getStatus(FuncRequest(LFUN_CHANGES_TRACK)).enabled() &&
|
bool const review =
|
||||||
lyx::getStatus(FuncRequest(LFUN_CHANGES_TRACK)).onoff(true);
|
lyx::getStatus(FuncRequest(LFUN_CHANGES_TRACK)).enabled() &&
|
||||||
|
lyx::getStatus(FuncRequest(LFUN_CHANGES_TRACK)).onoff(true);
|
||||||
|
|
||||||
|
toolbars_->update(math, table, review);
|
||||||
|
} else
|
||||||
|
toolbars_->update(false, false, false);
|
||||||
|
|
||||||
toolbars_->update(math, table, review);
|
|
||||||
// update redaonly status of open dialogs. This could also be in
|
// update redaonly status of open dialogs. This could also be in
|
||||||
// updateMenubar(), but since updateToolbars() and updateMenubar()
|
// updateMenubar(), but since updateToolbars() and updateMenubar()
|
||||||
// are always called together it is only here.
|
// are always called together it is only here.
|
||||||
@ -449,7 +362,7 @@ void LyXView::autoSave()
|
|||||||
{
|
{
|
||||||
LYXERR(Debug::INFO) << "Running autoSave()" << endl;
|
LYXERR(Debug::INFO) << "Running autoSave()" << endl;
|
||||||
|
|
||||||
if (view()->buffer())
|
if (buffer())
|
||||||
lyx::autoSave(view());
|
lyx::autoSave(view());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -464,7 +377,7 @@ void LyXView::resetAutosaveTimer()
|
|||||||
void LyXView::updateLayoutChoice()
|
void LyXView::updateLayoutChoice()
|
||||||
{
|
{
|
||||||
// Don't show any layouts without a buffer
|
// Don't show any layouts without a buffer
|
||||||
if (!view()->buffer()) {
|
if (!buffer()) {
|
||||||
toolbars_->clearLayoutList();
|
toolbars_->clearLayoutList();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -474,8 +387,7 @@ void LyXView::updateLayoutChoice()
|
|||||||
current_layout = buffer()->params().getTextClass().defaultLayoutName();
|
current_layout = buffer()->params().getTextClass().defaultLayoutName();
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_ASSERT(work_area_);
|
docstring const & layout = currentWorkArea()->bufferView().cursor().
|
||||||
docstring const & layout = work_area_->bufferView().cursor().
|
|
||||||
innerParagraph().layout()->name();
|
innerParagraph().layout()->name();
|
||||||
|
|
||||||
if (layout != current_layout) {
|
if (layout != current_layout) {
|
||||||
@ -490,42 +402,50 @@ void LyXView::updateWindowTitle()
|
|||||||
docstring maximize_title = lyx::from_ascii("LyX");
|
docstring maximize_title = lyx::from_ascii("LyX");
|
||||||
docstring minimize_title = lyx::from_ascii("LyX");
|
docstring minimize_title = lyx::from_ascii("LyX");
|
||||||
|
|
||||||
if (view()->buffer()) {
|
Buffer * buf = buffer();
|
||||||
string const cur_title = buffer()->fileName();
|
if (buf) {
|
||||||
|
string const cur_title = buf->fileName();
|
||||||
if (!cur_title.empty()) {
|
if (!cur_title.empty()) {
|
||||||
maximize_title += ": " + makeDisplayPath(cur_title, 30);
|
maximize_title += ": " + makeDisplayPath(cur_title, 30);
|
||||||
minimize_title = lyx::from_utf8(onlyFilename(cur_title));
|
minimize_title = lyx::from_utf8(onlyFilename(cur_title));
|
||||||
if (!buffer()->isClean()) {
|
if (!buf->isClean()) {
|
||||||
maximize_title += _(" (changed)");
|
maximize_title += _(" (changed)");
|
||||||
minimize_title += lyx::char_type('*');
|
minimize_title += lyx::char_type('*');
|
||||||
}
|
}
|
||||||
if (buffer()->isReadonly())
|
if (buf->isReadonly())
|
||||||
maximize_title += _(" (read only)");
|
maximize_title += _(" (read only)");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setWindowTitle(maximize_title, minimize_title);
|
setWindowTitle(maximize_title, minimize_title);
|
||||||
updateTab();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void LyXView::dispatch(FuncRequest const & cmd)
|
void LyXView::dispatch(FuncRequest const & cmd)
|
||||||
{
|
{
|
||||||
theLyXFunc().setLyXView(this);
|
string const argument = to_utf8(cmd.argument());
|
||||||
lyx::dispatch(cmd);
|
switch(cmd.action) {
|
||||||
|
case LFUN_BUFFER_SWITCH:
|
||||||
|
setBuffer(theBufferList().getBuffer(to_utf8(cmd.argument())));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
theLyXFunc().setLyXView(this);
|
||||||
|
lyx::dispatch(cmd);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Buffer const * const LyXView::updateInset(Inset const * inset) const
|
Buffer const * const LyXView::updateInset(Inset const * inset)
|
||||||
{
|
{
|
||||||
Buffer const * buffer_ptr = 0;
|
WorkArea * work_area = currentWorkArea();
|
||||||
if (inset) {
|
if (!work_area)
|
||||||
BOOST_ASSERT(work_area_);
|
return 0;
|
||||||
work_area_->scheduleRedraw();
|
|
||||||
|
|
||||||
buffer_ptr = work_area_->bufferView().buffer();
|
if (inset) {
|
||||||
|
BOOST_ASSERT(work_area);
|
||||||
|
work_area->scheduleRedraw();
|
||||||
}
|
}
|
||||||
return buffer_ptr;
|
return work_area->bufferView().buffer();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
|
@ -70,16 +70,19 @@ public:
|
|||||||
|
|
||||||
virtual void setFocus() = 0;
|
virtual void setFocus() = 0;
|
||||||
|
|
||||||
std::vector<int> const & workAreaIds() const { return work_area_ids_; }
|
///
|
||||||
|
virtual WorkArea * workArea(Buffer & buffer) = 0;
|
||||||
/// FIXME: rename to setCurrentWorkArea()
|
///
|
||||||
void setWorkArea(WorkArea * work_area);
|
virtual WorkArea * addWorkArea(Buffer & buffer) = 0;
|
||||||
|
///
|
||||||
|
virtual void setCurrentWorkArea(WorkArea * work_area) = 0;
|
||||||
|
///
|
||||||
|
virtual void removeWorkArea(WorkArea * work_area) = 0;
|
||||||
/// return the current WorkArea (the one that has the focus).
|
/// return the current WorkArea (the one that has the focus).
|
||||||
WorkArea const * currentWorkArea() const;
|
virtual WorkArea const * currentWorkArea() const = 0;
|
||||||
/// FIXME: This non-const access is needed because of
|
/// FIXME: This non-const access is needed because of
|
||||||
/// a mis-designed \c ControlSpellchecker.
|
/// a mis-designed \c ControlSpellchecker.
|
||||||
WorkArea * currentWorkArea();
|
virtual WorkArea * currentWorkArea() = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is called after the concrete view has been created.
|
* This is called after the concrete view has been created.
|
||||||
@ -114,14 +117,12 @@ public:
|
|||||||
|
|
||||||
//@{ generic accessor functions
|
//@{ generic accessor functions
|
||||||
|
|
||||||
/** return the current buffer view
|
/// \return the current buffer view.
|
||||||
Returned as a shared_ptr so that anything wanting to cache the
|
BufferView * view();
|
||||||
buffer view can do so safely using a boost::weak_ptr.
|
|
||||||
*/
|
|
||||||
BufferView * view() const;
|
|
||||||
|
|
||||||
/// return the buffer currently shown in this window
|
/// \return the buffer currently shown in this window
|
||||||
Buffer * buffer() const;
|
Buffer * buffer();
|
||||||
|
Buffer const * buffer() const;
|
||||||
|
|
||||||
/// return the toolbar for this view
|
/// return the toolbar for this view
|
||||||
Toolbars & getToolbars() { return *toolbars_.get(); }
|
Toolbars & getToolbars() { return *toolbars_.get(); }
|
||||||
@ -141,14 +142,11 @@ public:
|
|||||||
//@}
|
//@}
|
||||||
|
|
||||||
/// load a buffer into the current workarea.
|
/// load a buffer into the current workarea.
|
||||||
bool loadLyXFile(support::FileName const & name, ///< File to load.
|
Buffer * loadLyXFile(support::FileName const & name, ///< File to load.
|
||||||
bool tolastfiles = true, ///< append to the "Open recent" menu?
|
bool tolastfiles = true); ///< append to the "Open recent" menu?
|
||||||
bool child_document = false, ///< Is this a child document?
|
|
||||||
bool auto_open = false); ///< Is this being opened by LyX itself?
|
|
||||||
|
|
||||||
/// set a buffer to the current workarea.
|
/// set a buffer to the current workarea.
|
||||||
void setBuffer(Buffer * b, ///< \c Buffer to set.
|
void setBuffer(Buffer * b); ///< \c Buffer to set.
|
||||||
bool child_document = false); ///< Is this a child document?
|
|
||||||
|
|
||||||
/// updates the possible layouts selectable
|
/// updates the possible layouts selectable
|
||||||
void updateLayoutChoice();
|
void updateLayoutChoice();
|
||||||
@ -176,9 +174,6 @@ public:
|
|||||||
/// updates the title of the window
|
/// updates the title of the window
|
||||||
void updateWindowTitle();
|
void updateWindowTitle();
|
||||||
|
|
||||||
/// updates the tab view
|
|
||||||
virtual void updateTab() = 0;
|
|
||||||
|
|
||||||
/// reset autosave timer
|
/// reset autosave timer
|
||||||
void resetAutosaveTimer();
|
void resetAutosaveTimer();
|
||||||
|
|
||||||
@ -188,7 +183,7 @@ public:
|
|||||||
/** redraw \c inset in all the BufferViews in which it is currently
|
/** redraw \c inset in all the BufferViews in which it is currently
|
||||||
* visible. If successful return a pointer to the owning Buffer.
|
* visible. If successful return a pointer to the owning Buffer.
|
||||||
*/
|
*/
|
||||||
Buffer const * const updateInset(Inset const *) const;
|
Buffer const * const updateInset(Inset const *);
|
||||||
|
|
||||||
/// returns true if this view has the focus.
|
/// returns true if this view has the focus.
|
||||||
virtual bool hasFocus() const = 0;
|
virtual bool hasFocus() const = 0;
|
||||||
@ -196,17 +191,15 @@ public:
|
|||||||
/// show the error list to the user
|
/// show the error list to the user
|
||||||
void showErrorList(std::string const &);
|
void showErrorList(std::string const &);
|
||||||
|
|
||||||
|
protected:
|
||||||
/// connect to signals in the given BufferView
|
/// connect to signals in the given BufferView
|
||||||
void connectBufferView(BufferView & bv);
|
void connectBufferView(BufferView & bv);
|
||||||
/// disconnect from signals in the given BufferView
|
/// disconnect from signals in the given BufferView
|
||||||
void disconnectBufferView();
|
void disconnectBufferView();
|
||||||
|
/// connect to signals in the given buffer
|
||||||
protected:
|
void connectBuffer(Buffer & buf);
|
||||||
/// current work area (screen view of a BufferView).
|
/// disconnect from signals in the given buffer
|
||||||
/**
|
void disconnectBuffer();
|
||||||
\todo FIXME: there is only one workArea per LyXView for now.
|
|
||||||
*/
|
|
||||||
frontend::WorkArea * work_area_;
|
|
||||||
|
|
||||||
/// view's menubar
|
/// view's menubar
|
||||||
boost::scoped_ptr<Menubar> menubar_;
|
boost::scoped_ptr<Menubar> menubar_;
|
||||||
@ -231,8 +224,6 @@ private:
|
|||||||
/// dialogs for this view
|
/// dialogs for this view
|
||||||
boost::scoped_ptr<Dialogs> dialogs_;
|
boost::scoped_ptr<Dialogs> dialogs_;
|
||||||
|
|
||||||
/// buffer changed signal connection
|
|
||||||
boost::signals::connection bufferChangedConnection_;
|
|
||||||
/// buffer structure changed signal connection
|
/// buffer structure changed signal connection
|
||||||
boost::signals::connection bufferStructureChangedConnection_;
|
boost::signals::connection bufferStructureChangedConnection_;
|
||||||
/// buffer errors signal connection
|
/// buffer errors signal connection
|
||||||
@ -247,18 +238,6 @@ private:
|
|||||||
boost::signals::connection timerConnection_;
|
boost::signals::connection timerConnection_;
|
||||||
/// buffer readonly status changed signal connection
|
/// buffer readonly status changed signal connection
|
||||||
boost::signals::connection readonlyConnection_;
|
boost::signals::connection readonlyConnection_;
|
||||||
/// buffer closing signal connection
|
|
||||||
boost::signals::connection closingConnection_;
|
|
||||||
/// connect to signals in the given buffer
|
|
||||||
void connectBuffer(Buffer & buf);
|
|
||||||
/// disconnect from signals in the given buffer
|
|
||||||
/// NOTE: Do not call this unless you really want no buffer
|
|
||||||
/// to be connected---for example, when closing the last open
|
|
||||||
/// buffer. If you are switching buffers, just call
|
|
||||||
/// connectBuffer(), and the old buffer will be disconnected
|
|
||||||
/// automatically. This ensures that we do not leave LyX in a
|
|
||||||
/// state in which no buffer is connected.
|
|
||||||
void disconnectBuffer();
|
|
||||||
|
|
||||||
/// BufferView messages signal connection
|
/// BufferView messages signal connection
|
||||||
//@{
|
//@{
|
||||||
@ -298,7 +277,6 @@ protected:
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
int id_;
|
int id_;
|
||||||
std::vector<int> work_area_ids_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
|
@ -17,26 +17,27 @@
|
|||||||
|
|
||||||
#include "frontends/Application.h"
|
#include "frontends/Application.h"
|
||||||
#include "frontends/FontMetrics.h"
|
#include "frontends/FontMetrics.h"
|
||||||
|
#include "frontends/LyXView.h"
|
||||||
#include "FuncRequest.h"
|
|
||||||
#include "LyXFunc.h"
|
|
||||||
|
|
||||||
#include "BufferView.h"
|
#include "BufferView.h"
|
||||||
#include "Buffer.h"
|
#include "Buffer.h"
|
||||||
#include "BufferParams.h"
|
#include "BufferParams.h"
|
||||||
|
#include "Color.h"
|
||||||
#include "CoordCache.h"
|
#include "CoordCache.h"
|
||||||
#include "Cursor.h"
|
#include "Cursor.h"
|
||||||
#include "debug.h"
|
#include "debug.h"
|
||||||
#include "Language.h"
|
|
||||||
#include "Color.h"
|
|
||||||
#include "Font.h"
|
#include "Font.h"
|
||||||
|
#include "FuncRequest.h"
|
||||||
|
#include "Language.h"
|
||||||
|
#include "LyX.h"
|
||||||
|
#include "LyXFunc.h"
|
||||||
#include "LyXRC.h"
|
#include "LyXRC.h"
|
||||||
#include "Text.h"
|
|
||||||
#include "LyXView.h"
|
|
||||||
#include "MetricsInfo.h"
|
#include "MetricsInfo.h"
|
||||||
|
#include "Text.h"
|
||||||
|
|
||||||
#include "gettext.h"
|
#include "gettext.h"
|
||||||
#include "support/ForkedcallsController.h"
|
#include "support/ForkedcallsController.h"
|
||||||
|
#include "support/FileName.h"
|
||||||
|
|
||||||
#include <boost/utility.hpp>
|
#include <boost/utility.hpp>
|
||||||
#include <boost/bind.hpp>
|
#include <boost/bind.hpp>
|
||||||
@ -64,41 +65,56 @@ boost::signals::connection timecon;
|
|||||||
namespace lyx {
|
namespace lyx {
|
||||||
namespace frontend {
|
namespace frontend {
|
||||||
|
|
||||||
WorkArea::WorkArea(int id, LyXView & lyx_view)
|
WorkArea::WorkArea(Buffer & buffer, LyXView & lv)
|
||||||
: buffer_view_(0), lyx_view_(lyx_view), greyed_out_(true),
|
: buffer_view_(new BufferView(buffer)), lyx_view_(&lv),
|
||||||
id_(id), cursor_visible_(false), cursor_timeout_(400)
|
cursor_visible_(false), cursor_timeout_(400)
|
||||||
{
|
{
|
||||||
// Start loading the pixmap as soon as possible
|
|
||||||
//if (lyxrc.show_banner) {
|
|
||||||
// showBanner();
|
|
||||||
//}
|
|
||||||
|
|
||||||
// Setup the signals
|
// Setup the signals
|
||||||
timecon = cursor_timeout_.timeout
|
timecon = cursor_timeout_.timeout
|
||||||
.connect(boost::bind(&WorkArea::toggleCursor, this));
|
.connect(boost::bind(&WorkArea::toggleCursor, this));
|
||||||
|
|
||||||
|
bufferChangedConnection_ =
|
||||||
|
buffer.changed.connect(
|
||||||
|
boost::bind(&WorkArea::redraw, this));
|
||||||
|
|
||||||
|
bufferClosingConnection_ =
|
||||||
|
buffer.closing.connect(
|
||||||
|
boost::bind(&WorkArea::close, this));
|
||||||
|
|
||||||
cursor_timeout_.start();
|
cursor_timeout_.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void WorkArea::setBufferView(BufferView * buffer_view)
|
WorkArea::~WorkArea()
|
||||||
{
|
{
|
||||||
if (buffer_view_) {
|
bufferChangedConnection_.disconnect();
|
||||||
message_connection_.disconnect();
|
bufferClosingConnection_.disconnect();
|
||||||
lyx_view_.disconnectBufferView();
|
|
||||||
}
|
|
||||||
|
|
||||||
hideCursor();
|
// current buffer is going to be switched-off, save cursor pos
|
||||||
buffer_view_ = buffer_view;
|
// Ideally, the whole cursor stack should be saved, but session
|
||||||
toggleCursor();
|
// currently can only handle bottom (whole document) level pit and pos.
|
||||||
|
// That is to say, if a cursor is in a nested inset, it will be
|
||||||
|
// restore to the left of the top level inset.
|
||||||
|
Cursor & cur = buffer_view_->cursor();
|
||||||
|
LyX::ref().session().lastFilePos().save(
|
||||||
|
support::FileName(buffer_view_->buffer()->fileName()),
|
||||||
|
boost::tie(cur.bottom().pit(), cur.bottom().pos()) );
|
||||||
|
|
||||||
message_connection_ = buffer_view_->message.connect(
|
delete buffer_view_;
|
||||||
boost::bind(&WorkArea::displayMessage, this, _1));
|
|
||||||
|
|
||||||
lyx_view_.connectBufferView(*buffer_view);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void WorkArea::close()
|
||||||
|
{
|
||||||
|
lyx_view_->removeWorkArea(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
//void WorkArea::setLyXView(LyXView * lyx_view)
|
||||||
|
//{
|
||||||
|
// lyx_view_ = lyx_view;
|
||||||
|
//}
|
||||||
|
|
||||||
|
|
||||||
BufferView & WorkArea::bufferView()
|
BufferView & WorkArea::bufferView()
|
||||||
{
|
{
|
||||||
return *buffer_view_;
|
return *buffer_view_;
|
||||||
@ -127,16 +143,13 @@ void WorkArea::startBlinkingCursor()
|
|||||||
|
|
||||||
void WorkArea::redraw()
|
void WorkArea::redraw()
|
||||||
{
|
{
|
||||||
if (!buffer_view_ || !buffer_view_->buffer()) {
|
if (!isVisible())
|
||||||
greyed_out_ = true;
|
// No need to redraw in this case.
|
||||||
// The argument here are meaningless.
|
|
||||||
expose(1,1,1,1);
|
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
// No need to do anything if this is the current view. The BufferView
|
// No need to do anything if this is the current view. The BufferView
|
||||||
// metrics are already up to date.
|
// metrics are already up to date.
|
||||||
if (&lyx_view_ != theApp()->currentView()) {
|
if (lyx_view_ != theApp()->currentView()) {
|
||||||
// FIXME: it would be nice to optimize for the off-screen case.
|
// FIXME: it would be nice to optimize for the off-screen case.
|
||||||
buffer_view_->updateMetrics(false);
|
buffer_view_->updateMetrics(false);
|
||||||
buffer_view_->cursor().fixIfBroken();
|
buffer_view_->cursor().fixIfBroken();
|
||||||
@ -152,7 +165,6 @@ void WorkArea::redraw()
|
|||||||
}
|
}
|
||||||
|
|
||||||
ViewMetricsInfo const & vi = buffer_view_->viewMetricsInfo();
|
ViewMetricsInfo const & vi = buffer_view_->viewMetricsInfo();
|
||||||
greyed_out_ = false;
|
|
||||||
|
|
||||||
LYXERR(Debug::WORKAREA) << "WorkArea::redraw screen" << endl;
|
LYXERR(Debug::WORKAREA) << "WorkArea::redraw screen" << endl;
|
||||||
|
|
||||||
@ -176,13 +188,8 @@ void WorkArea::processKeySym(KeySymbolPtr key, key_modifier::state state)
|
|||||||
// the blinking cursor.
|
// the blinking cursor.
|
||||||
stopBlinkingCursor();
|
stopBlinkingCursor();
|
||||||
|
|
||||||
theLyXFunc().setLyXView(&lyx_view_);
|
theLyXFunc().setLyXView(lyx_view_);
|
||||||
theLyXFunc().processKeySym(key, state);
|
theLyXFunc().processKeySym(key, state);
|
||||||
|
|
||||||
/* When we move around, or type, it's nice to be able to see
|
|
||||||
* the cursor immediately after the keypress.
|
|
||||||
*/
|
|
||||||
startBlinkingCursor();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -190,11 +197,11 @@ void WorkArea::dispatch(FuncRequest const & cmd0, key_modifier::state k)
|
|||||||
{
|
{
|
||||||
// Handle drag&drop
|
// Handle drag&drop
|
||||||
if (cmd0.action == LFUN_FILE_OPEN) {
|
if (cmd0.action == LFUN_FILE_OPEN) {
|
||||||
lyx_view_.dispatch(cmd0);
|
lyx_view_->dispatch(cmd0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
theLyXFunc().setLyXView(&lyx_view_);
|
theLyXFunc().setLyXView(lyx_view_);
|
||||||
|
|
||||||
FuncRequest cmd;
|
FuncRequest cmd;
|
||||||
|
|
||||||
@ -222,9 +229,9 @@ void WorkArea::dispatch(FuncRequest const & cmd0, key_modifier::state k)
|
|||||||
|
|
||||||
// Skip these when selecting
|
// Skip these when selecting
|
||||||
if (cmd.action != LFUN_MOUSE_MOTION) {
|
if (cmd.action != LFUN_MOUSE_MOTION) {
|
||||||
lyx_view_.updateLayoutChoice();
|
lyx_view_->updateLayoutChoice();
|
||||||
lyx_view_.updateMenubar();
|
lyx_view_->updateMenubar();
|
||||||
lyx_view_.updateToolbars();
|
lyx_view_->updateToolbars();
|
||||||
}
|
}
|
||||||
|
|
||||||
// GUI tweaks except with mouse motion with no button pressed.
|
// GUI tweaks except with mouse motion with no button pressed.
|
||||||
@ -233,7 +240,7 @@ void WorkArea::dispatch(FuncRequest const & cmd0, key_modifier::state k)
|
|||||||
// Slight hack: this is only called currently when we
|
// Slight hack: this is only called currently when we
|
||||||
// clicked somewhere, so we force through the display
|
// clicked somewhere, so we force through the display
|
||||||
// of the new status here.
|
// of the new status here.
|
||||||
lyx_view_.clearMessage();
|
lyx_view_->clearMessage();
|
||||||
|
|
||||||
// Show the cursor immediately after any operation.
|
// Show the cursor immediately after any operation.
|
||||||
startBlinkingCursor();
|
startBlinkingCursor();
|
||||||
@ -243,12 +250,12 @@ void WorkArea::dispatch(FuncRequest const & cmd0, key_modifier::state k)
|
|||||||
|
|
||||||
void WorkArea::resizeBufferView()
|
void WorkArea::resizeBufferView()
|
||||||
{
|
{
|
||||||
lyx_view_.busy(true);
|
lyx_view_->busy(true);
|
||||||
lyx_view_.message(_("Formatting document..."));
|
lyx_view_->message(_("Formatting document..."));
|
||||||
buffer_view_->workAreaResize(width(), height());
|
buffer_view_->workAreaResize(width(), height());
|
||||||
lyx_view_.updateLayoutChoice();
|
lyx_view_->updateLayoutChoice();
|
||||||
lyx_view_.clearMessage();
|
lyx_view_->clearMessage();
|
||||||
lyx_view_.busy(false);
|
lyx_view_->busy(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -268,7 +275,7 @@ void WorkArea::scrollBufferView(int position)
|
|||||||
redraw();
|
redraw();
|
||||||
if (lyxrc.cursor_follows_scrollbar) {
|
if (lyxrc.cursor_follows_scrollbar) {
|
||||||
buffer_view_->setCursorFromScrollbar();
|
buffer_view_->setCursorFromScrollbar();
|
||||||
lyx_view_.updateLayoutChoice();
|
lyx_view_->updateLayoutChoice();
|
||||||
}
|
}
|
||||||
// Show the cursor immediately after any operation.
|
// Show the cursor immediately after any operation.
|
||||||
startBlinkingCursor();
|
startBlinkingCursor();
|
||||||
@ -280,9 +287,6 @@ void WorkArea::showCursor()
|
|||||||
if (cursor_visible_)
|
if (cursor_visible_)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (!buffer_view_->buffer())
|
|
||||||
return;
|
|
||||||
|
|
||||||
CursorShape shape = BAR_SHAPE;
|
CursorShape shape = BAR_SHAPE;
|
||||||
|
|
||||||
Text const & text = *buffer_view_->cursor().innerText();
|
Text const & text = *buffer_view_->cursor().innerText();
|
||||||
@ -332,28 +336,19 @@ void WorkArea::hideCursor()
|
|||||||
|
|
||||||
void WorkArea::toggleCursor()
|
void WorkArea::toggleCursor()
|
||||||
{
|
{
|
||||||
if (buffer_view_->buffer()) {
|
if (cursor_visible_)
|
||||||
|
hideCursor();
|
||||||
|
else
|
||||||
|
showCursor();
|
||||||
|
|
||||||
if (cursor_visible_)
|
// Use this opportunity to deal with any child processes that
|
||||||
hideCursor();
|
// have finished but are waiting to communicate this fact
|
||||||
else
|
// to the rest of LyX.
|
||||||
showCursor();
|
ForkedcallsController & fcc = ForkedcallsController::get();
|
||||||
|
fcc.handleCompletedProcesses();
|
||||||
// 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();
|
cursor_timeout_.restart();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void WorkArea::displayMessage(lyx::docstring const & message)
|
|
||||||
{
|
|
||||||
lyx_view_.message(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
} // namespace lyx
|
} // namespace lyx
|
||||||
|
@ -25,7 +25,7 @@
|
|||||||
#undef CursorShape
|
#undef CursorShape
|
||||||
|
|
||||||
namespace lyx {
|
namespace lyx {
|
||||||
|
class Buffer;
|
||||||
class BufferView;
|
class BufferView;
|
||||||
class FuncRequest;
|
class FuncRequest;
|
||||||
|
|
||||||
@ -53,13 +53,13 @@ enum CursorShape {
|
|||||||
*/
|
*/
|
||||||
class WorkArea : public boost::signals::trackable {
|
class WorkArea : public boost::signals::trackable {
|
||||||
public:
|
public:
|
||||||
WorkArea(int id, LyXView & lyx_view);
|
///
|
||||||
|
WorkArea(Buffer & buffer, LyXView & lv);
|
||||||
|
|
||||||
virtual ~WorkArea() {}
|
virtual ~WorkArea();
|
||||||
|
|
||||||
int const id() const { return id_; }
|
///
|
||||||
|
void setLyXView(LyXView & lv) { lyx_view_ = &lv; }
|
||||||
void setBufferView(BufferView * buffer_view);
|
|
||||||
|
|
||||||
///
|
///
|
||||||
BufferView & bufferView();
|
BufferView & bufferView();
|
||||||
@ -69,6 +69,9 @@ public:
|
|||||||
/// \return true if has the keyboard input focus.
|
/// \return true if has the keyboard input focus.
|
||||||
virtual bool hasFocus() const = 0;
|
virtual bool hasFocus() const = 0;
|
||||||
|
|
||||||
|
/// \return true if has this WorkArea is visible.
|
||||||
|
virtual bool isVisible() const = 0;
|
||||||
|
|
||||||
/// return the width of the work area in pixels
|
/// return the width of the work area in pixels
|
||||||
virtual int width() const = 0;
|
virtual int width() const = 0;
|
||||||
|
|
||||||
@ -95,12 +98,17 @@ public:
|
|||||||
/// Process Key pressed event.
|
/// Process Key pressed event.
|
||||||
/// This needs to be public because it is accessed externally by GuiView.
|
/// This needs to be public because it is accessed externally by GuiView.
|
||||||
void processKeySym(KeySymbolPtr key, key_modifier::state state);
|
void processKeySym(KeySymbolPtr key, key_modifier::state state);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/// cause the display of the given area of the work area
|
/// cause the display of the given area of the work area
|
||||||
virtual void expose(int x, int y, int w, int h) = 0;
|
virtual void expose(int x, int y, int w, int h) = 0;
|
||||||
///
|
///
|
||||||
void dispatch(FuncRequest const & cmd0,
|
void dispatch(FuncRequest const & cmd0,
|
||||||
key_modifier::state = key_modifier::none);
|
key_modifier::state = key_modifier::none);
|
||||||
|
|
||||||
|
/// close this work area.
|
||||||
|
/// Slot for Buffer::closing boost signal.
|
||||||
|
void close();
|
||||||
///
|
///
|
||||||
void resizeBufferView();
|
void resizeBufferView();
|
||||||
///
|
///
|
||||||
@ -120,25 +128,20 @@ protected:
|
|||||||
|
|
||||||
///
|
///
|
||||||
BufferView * buffer_view_;
|
BufferView * buffer_view_;
|
||||||
|
|
||||||
///
|
///
|
||||||
LyXView & lyx_view_;
|
LyXView * lyx_view_;
|
||||||
///
|
|
||||||
bool greyed_out_;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
///
|
|
||||||
int id_;
|
|
||||||
///
|
|
||||||
void displayMessage(docstring const &);
|
|
||||||
/// buffer messages signal connection
|
|
||||||
boost::signals::connection message_connection_;
|
|
||||||
|
|
||||||
/// is the cursor currently displayed
|
/// is the cursor currently displayed
|
||||||
bool cursor_visible_;
|
bool cursor_visible_;
|
||||||
|
|
||||||
///
|
///
|
||||||
Timeout cursor_timeout_;
|
Timeout cursor_timeout_;
|
||||||
|
|
||||||
|
/// buffer changed signal connection
|
||||||
|
boost::signals::connection bufferChangedConnection_;
|
||||||
|
/// buffer closing signal connection
|
||||||
|
boost::signals::connection bufferClosingConnection_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
|
@ -42,6 +42,7 @@
|
|||||||
#include <QFileOpenEvent>
|
#include <QFileOpenEvent>
|
||||||
#include <QLocale>
|
#include <QLocale>
|
||||||
#include <QLibraryInfo>
|
#include <QLibraryInfo>
|
||||||
|
#include <QPixmapCache>
|
||||||
#include <QTextCodec>
|
#include <QTextCodec>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
#include <QTranslator>
|
#include <QTranslator>
|
||||||
@ -147,6 +148,10 @@ GuiApplication::GuiApplication(int & argc, char ** argv)
|
|||||||
LoaderQueue::setPriority(10,100);
|
LoaderQueue::setPriority(10,100);
|
||||||
|
|
||||||
guiApp = this;
|
guiApp = this;
|
||||||
|
|
||||||
|
// Set the cache to 5120 kilobytes which corresponds to screen size of
|
||||||
|
// 1280 by 1024 pixels with a color depth of 32 bits.
|
||||||
|
QPixmapCache::setCacheLimit(5120);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -12,22 +12,12 @@
|
|||||||
|
|
||||||
#include <config.h>
|
#include <config.h>
|
||||||
|
|
||||||
// This include must be declared before everything else because
|
#include "GuiImplementation.h"
|
||||||
// of boost/Qt/LyX clash...
|
|
||||||
#include "GuiView.h"
|
#include "GuiView.h"
|
||||||
|
|
||||||
#include "GuiImplementation.h"
|
|
||||||
#include "GuiWorkArea.h"
|
|
||||||
|
|
||||||
#include "BufferView.h"
|
|
||||||
#include "BufferList.h"
|
|
||||||
#include "FuncRequest.h"
|
|
||||||
#include "LyXFunc.h"
|
|
||||||
|
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
|
|
||||||
using boost::shared_ptr;
|
|
||||||
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
@ -49,7 +39,6 @@ namespace frontend {
|
|||||||
GuiImplementation::GuiImplementation()
|
GuiImplementation::GuiImplementation()
|
||||||
{
|
{
|
||||||
view_ids_.clear();
|
view_ids_.clear();
|
||||||
work_area_ids_.clear();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -74,9 +63,6 @@ bool GuiImplementation::unregisterView(int id)
|
|||||||
std::map<int, GuiView *>::iterator it;
|
std::map<int, GuiView *>::iterator it;
|
||||||
for (it = views_.begin(); it != views_.end(); ++it) {
|
for (it = views_.begin(); it != views_.end(); ++it) {
|
||||||
if (it->first == id) {
|
if (it->first == id) {
|
||||||
std::vector<int> const & wa_ids = it->second->workAreaIds();
|
|
||||||
for (size_t i = 0; i < wa_ids.size(); ++i)
|
|
||||||
work_areas_.erase(wa_ids[i]);
|
|
||||||
views_.erase(id);
|
views_.erase(id);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -108,9 +94,7 @@ bool GuiImplementation::closeAllViews()
|
|||||||
}
|
}
|
||||||
|
|
||||||
views_.clear();
|
views_.clear();
|
||||||
work_areas_.clear();
|
|
||||||
view_ids_.clear();
|
view_ids_.clear();
|
||||||
work_area_ids_.clear();
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,43 +106,6 @@ LyXView& GuiImplementation::view(int id) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
std::vector<int> const & GuiImplementation::workAreaIds()
|
|
||||||
{
|
|
||||||
updateIds(work_areas_, work_area_ids_);
|
|
||||||
return work_area_ids_;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int GuiImplementation::newWorkArea(unsigned int w, unsigned int h, int view_id)
|
|
||||||
{
|
|
||||||
updateIds(views_, view_ids_);
|
|
||||||
int id = 0;
|
|
||||||
while (work_areas_.find(id) != work_areas_.end())
|
|
||||||
id++;
|
|
||||||
|
|
||||||
GuiView * view = views_[view_id];
|
|
||||||
|
|
||||||
work_areas_.insert(std::pair<int, GuiWorkArea *>
|
|
||||||
(id, new GuiWorkArea(w, h, id, *view)));
|
|
||||||
|
|
||||||
// FIXME BufferView creation should be independant of WorkArea creation
|
|
||||||
buffer_views_[id].reset(new BufferView);
|
|
||||||
work_areas_[id]->setBufferView(buffer_views_[id].get());
|
|
||||||
|
|
||||||
view->setWorkArea(work_areas_[id]);
|
|
||||||
view->initTab(work_areas_[id]);
|
|
||||||
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
WorkArea& GuiImplementation::workArea(int id)
|
|
||||||
{
|
|
||||||
BOOST_ASSERT(work_areas_.find(id) != work_areas_.end());
|
|
||||||
return *work_areas_[id];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
} // namespace lyx
|
} // namespace lyx
|
||||||
|
|
||||||
|
@ -22,7 +22,6 @@
|
|||||||
namespace lyx {
|
namespace lyx {
|
||||||
namespace frontend {
|
namespace frontend {
|
||||||
|
|
||||||
class GuiWorkArea;
|
|
||||||
class GuiView;
|
class GuiView;
|
||||||
class LyXView;
|
class LyXView;
|
||||||
|
|
||||||
@ -44,9 +43,6 @@ public:
|
|||||||
|
|
||||||
virtual LyXView& view(int id) const;
|
virtual LyXView& view(int id) const;
|
||||||
|
|
||||||
virtual int newWorkArea(unsigned int width, unsigned int height, int view_id);
|
|
||||||
virtual WorkArea& workArea(int id);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
/// Multiple views container.
|
/// Multiple views container.
|
||||||
@ -56,22 +52,6 @@ private:
|
|||||||
* \sa Qt::WA_DeleteOnClose attribute.
|
* \sa Qt::WA_DeleteOnClose attribute.
|
||||||
*/
|
*/
|
||||||
std::map<int, GuiView *> views_;
|
std::map<int, GuiView *> views_;
|
||||||
|
|
||||||
/// Multiple workareas container.
|
|
||||||
/**
|
|
||||||
* Warning: This must not be a smart pointer as the destruction of the
|
|
||||||
* object is handled by Qt when its parent view is closed.
|
|
||||||
*/
|
|
||||||
std::map<int, GuiWorkArea *> work_areas_;
|
|
||||||
///
|
|
||||||
|
|
||||||
/// view of a buffer. Eventually there will be several.
|
|
||||||
std::map<int, boost::shared_ptr<BufferView> > buffer_views_;
|
|
||||||
|
|
||||||
|
|
||||||
std::vector<int> const & workAreaIds();
|
|
||||||
|
|
||||||
std::vector<int> work_area_ids_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
#include "qt_helpers.h"
|
#include "qt_helpers.h"
|
||||||
|
|
||||||
#include "frontends/Application.h"
|
#include "frontends/Application.h"
|
||||||
|
#include "frontends/Dialogs.h"
|
||||||
#include "frontends/Gui.h"
|
#include "frontends/Gui.h"
|
||||||
#include "frontends/WorkArea.h"
|
#include "frontends/WorkArea.h"
|
||||||
|
|
||||||
@ -43,6 +44,7 @@
|
|||||||
#include "LyXRC.h"
|
#include "LyXRC.h"
|
||||||
#include "MenuBackend.h"
|
#include "MenuBackend.h"
|
||||||
#include "Session.h"
|
#include "Session.h"
|
||||||
|
#include "version.h"
|
||||||
|
|
||||||
#include <QAction>
|
#include <QAction>
|
||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
@ -50,17 +52,16 @@
|
|||||||
#include <QDesktopWidget>
|
#include <QDesktopWidget>
|
||||||
#include <QDragEnterEvent>
|
#include <QDragEnterEvent>
|
||||||
#include <QDropEvent>
|
#include <QDropEvent>
|
||||||
#include <QHBoxLayout>
|
|
||||||
#include <QList>
|
#include <QList>
|
||||||
#include <QMimeData>
|
#include <QMimeData>
|
||||||
|
#include <QPainter>
|
||||||
#include <QPixmap>
|
#include <QPixmap>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
|
#include <QStackedWidget>
|
||||||
#include <QStatusBar>
|
#include <QStatusBar>
|
||||||
#include <QToolBar>
|
#include <QToolBar>
|
||||||
#include <QTabBar>
|
#include <QTabWidget>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
#include <QVBoxLayout>
|
|
||||||
|
|
||||||
|
|
||||||
#include <boost/bind.hpp>
|
#include <boost/bind.hpp>
|
||||||
#include <boost/shared_ptr.hpp>
|
#include <boost/shared_ptr.hpp>
|
||||||
@ -81,84 +82,59 @@ namespace {
|
|||||||
|
|
||||||
int const statusbar_timer_value = 3000;
|
int const statusbar_timer_value = 3000;
|
||||||
|
|
||||||
class TabWidget : public QWidget
|
class BackgroundWidget: public QWidget
|
||||||
{
|
{
|
||||||
QHBoxLayout* hlayout;
|
|
||||||
public:
|
public:
|
||||||
QTabBar* tabbar;
|
BackgroundWidget(QString const & file, QString const & text)
|
||||||
QPushButton* closeTabButton;
|
|
||||||
|
|
||||||
void hideTabsIfNecessary()
|
|
||||||
{
|
{
|
||||||
if (tabbar->count() > 1) {
|
splash_ = new QPixmap(file);
|
||||||
tabbar->show();
|
if (!splash_) {
|
||||||
closeTabButton->show();
|
lyxerr << "could not load splash screen: '" << fromqstr(file) << "'" << endl;
|
||||||
} else {
|
return;
|
||||||
tabbar->hide();
|
|
||||||
closeTabButton->hide();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QPainter pain(splash_);
|
||||||
|
pain.setPen(QColor(255, 255, 0));
|
||||||
|
QFont font;
|
||||||
|
// The font used to display the version info
|
||||||
|
font.setStyleHint(QFont::SansSerif);
|
||||||
|
font.setWeight(QFont::Bold);
|
||||||
|
font.setPointSize(convert<int>(lyxrc.font_sizes[Font::SIZE_LARGE]));
|
||||||
|
pain.setFont(font);
|
||||||
|
pain.drawText(260, 270, text);
|
||||||
}
|
}
|
||||||
|
|
||||||
TabWidget(QWidget* w, bool topTabBar)
|
void paintEvent(QPaintEvent * ev)
|
||||||
{
|
{
|
||||||
closeTabButton = new QPushButton(this);
|
if (!splash_)
|
||||||
FileName const file = support::libFileSearch("images", "closetab", "xpm");
|
return;
|
||||||
if (!file.empty()) {
|
|
||||||
QPixmap pm(toqstr(file.absFilename()));
|
|
||||||
closeTabButton->setIcon(QIcon(pm));
|
|
||||||
closeTabButton->setMaximumSize(pm.size());
|
|
||||||
closeTabButton->setFlat(true);
|
|
||||||
} else {
|
|
||||||
closeTabButton->setText("Close");
|
|
||||||
}
|
|
||||||
closeTabButton->setCursor(Qt::ArrowCursor);
|
|
||||||
closeTabButton->setToolTip(tr("Close tab"));
|
|
||||||
closeTabButton->setEnabled(true);
|
|
||||||
|
|
||||||
tabbar = new QTabBar;
|
int x = (width() - splash_->width()) / 2;
|
||||||
#if QT_VERSION >= 0x040200
|
int y = (height() - splash_->height()) / 2;
|
||||||
tabbar->setUsesScrollButtons(true);
|
QPainter pain(this);
|
||||||
#endif
|
pain.drawPixmap(x, y, *splash_);
|
||||||
hlayout = new QHBoxLayout;
|
|
||||||
QVBoxLayout* vlayout = new QVBoxLayout;
|
|
||||||
hlayout->addWidget(tabbar);
|
|
||||||
hlayout->addWidget(closeTabButton);
|
|
||||||
if (topTabBar) {
|
|
||||||
vlayout->addLayout(hlayout);
|
|
||||||
vlayout->addWidget(w);
|
|
||||||
} else {
|
|
||||||
tabbar->setShape(QTabBar::RoundedSouth);
|
|
||||||
vlayout->addWidget(w);
|
|
||||||
vlayout->addLayout(hlayout);
|
|
||||||
}
|
|
||||||
vlayout->setMargin(0);
|
|
||||||
vlayout->setSpacing(0);
|
|
||||||
hlayout->setMargin(0);
|
|
||||||
setLayout(vlayout);
|
|
||||||
hideTabsIfNecessary();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void clearTabbar()
|
private:
|
||||||
{
|
QPixmap * splash_;
|
||||||
for (int i = tabbar->count() - 1; i >= 0; --i)
|
|
||||||
tabbar->removeTab(i);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
} // namespace anon
|
} // namespace anon
|
||||||
|
|
||||||
|
|
||||||
struct GuiView::GuiViewPrivate
|
struct GuiView::GuiViewPrivate
|
||||||
{
|
{
|
||||||
vector<string> tabnames;
|
|
||||||
string cur_title;
|
string cur_title;
|
||||||
|
|
||||||
TabWidget* tabWidget;
|
|
||||||
|
|
||||||
int posx_offset;
|
int posx_offset;
|
||||||
int posy_offset;
|
int posy_offset;
|
||||||
|
|
||||||
GuiViewPrivate() : tabWidget(0), posx_offset(0), posy_offset(0)
|
QTabWidget * tab_widget_;
|
||||||
|
QStackedWidget * stack_widget_;
|
||||||
|
BackgroundWidget * bg_widget_;
|
||||||
|
|
||||||
|
GuiViewPrivate() : posx_offset(0), posy_offset(0)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
unsigned int smallIconSize;
|
unsigned int smallIconSize;
|
||||||
@ -201,6 +177,28 @@ struct GuiView::GuiViewPrivate
|
|||||||
|
|
||||||
return menu;
|
return menu;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void initBackground()
|
||||||
|
{
|
||||||
|
bg_widget_ = 0;
|
||||||
|
LYXERR(Debug::GUI) << "show banner: " << lyxrc.show_banner << endl;
|
||||||
|
/// The text to be written on top of the pixmap
|
||||||
|
QString const text = lyx_version ? QString(lyx_version) : qt_("unknown version");
|
||||||
|
FileName const file = support::libFileSearch("images", "banner", "png");
|
||||||
|
if (file.empty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
bg_widget_ = new BackgroundWidget(toqstr(file.absFilename()), text);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setBackground()
|
||||||
|
{
|
||||||
|
if (!bg_widget_)
|
||||||
|
return;
|
||||||
|
|
||||||
|
stack_widget_->setCurrentWidget(bg_widget_);
|
||||||
|
bg_widget_->setUpdatesEnabled(true);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -228,6 +226,44 @@ GuiView::GuiView(int id)
|
|||||||
setWindowIcon(QPixmap(toqstr(iconname.absFilename())));
|
setWindowIcon(QPixmap(toqstr(iconname.absFilename())));
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
d.tab_widget_ = new QTabWidget;
|
||||||
|
|
||||||
|
QPushButton * closeTabButton = new QPushButton(this);
|
||||||
|
FileName const file = support::libFileSearch("images", "closetab", "xpm");
|
||||||
|
if (!file.empty()) {
|
||||||
|
QPixmap pm(toqstr(file.absFilename()));
|
||||||
|
closeTabButton->setIcon(QIcon(pm));
|
||||||
|
closeTabButton->setMaximumSize(pm.size());
|
||||||
|
closeTabButton->setFlat(true);
|
||||||
|
} else {
|
||||||
|
closeTabButton->setText("Close");
|
||||||
|
}
|
||||||
|
closeTabButton->setCursor(Qt::ArrowCursor);
|
||||||
|
closeTabButton->setToolTip(tr("Close tab"));
|
||||||
|
closeTabButton->setEnabled(true);
|
||||||
|
|
||||||
|
QObject::connect(d.tab_widget_, SIGNAL(currentChanged(int)),
|
||||||
|
this, SLOT(currentTabChanged(int)));
|
||||||
|
QObject::connect(closeTabButton, SIGNAL(clicked()),
|
||||||
|
this, SLOT(closeCurrentTab()));
|
||||||
|
|
||||||
|
d.tab_widget_->setCornerWidget(closeTabButton);
|
||||||
|
#if QT_VERSION >= 0x040200
|
||||||
|
d.tab_widget_->setUsesScrollButtons(true);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
d.initBackground();
|
||||||
|
if (d.bg_widget_) {
|
||||||
|
lyxerr << "stack widget!" << endl;
|
||||||
|
d.stack_widget_ = new QStackedWidget;
|
||||||
|
d.stack_widget_->addWidget(d.bg_widget_);
|
||||||
|
d.stack_widget_->addWidget(d.tab_widget_);
|
||||||
|
setCentralWidget(d.stack_widget_);
|
||||||
|
} else {
|
||||||
|
d.stack_widget_ = 0;
|
||||||
|
setCentralWidget(d.tab_widget_);
|
||||||
|
}
|
||||||
|
|
||||||
// For Drag&Drop.
|
// For Drag&Drop.
|
||||||
setAcceptDrops(true);
|
setAcceptDrops(true);
|
||||||
}
|
}
|
||||||
@ -250,8 +286,8 @@ void GuiView::close()
|
|||||||
|
|
||||||
void GuiView::setFocus()
|
void GuiView::setFocus()
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(work_area_);
|
if (d.tab_widget_->count())
|
||||||
static_cast<GuiWorkArea *>(work_area_)->setFocus();
|
d.tab_widget_->currentWidget()->setFocus();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -274,14 +310,8 @@ void GuiView::init()
|
|||||||
QObject::connect(&statusbar_timer_, SIGNAL(timeout()),
|
QObject::connect(&statusbar_timer_, SIGNAL(timeout()),
|
||||||
this, SLOT(update_view_state_qt()));
|
this, SLOT(update_view_state_qt()));
|
||||||
|
|
||||||
BOOST_ASSERT(work_area_);
|
if (d.stack_widget_)
|
||||||
if (!work_area_->bufferView().buffer() && !theBufferList().empty())
|
d.stack_widget_->setCurrentWidget(d.bg_widget_);
|
||||||
setBuffer(theBufferList().first());
|
|
||||||
|
|
||||||
// make sure the buttons are disabled if needed
|
|
||||||
updateToolbars();
|
|
||||||
updateLayoutChoice();
|
|
||||||
updateMenubar();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -311,15 +341,6 @@ void GuiView::closeEvent(QCloseEvent * close_event)
|
|||||||
|
|
||||||
quitting = true;
|
quitting = true;
|
||||||
|
|
||||||
if (view()->buffer()) {
|
|
||||||
// save cursor position for opened files to .lyx/session
|
|
||||||
// only bottom (whole doc) level pit and pos is saved.
|
|
||||||
LyX::ref().session().lastFilePos().save(
|
|
||||||
FileName(buffer()->fileName()),
|
|
||||||
boost::tie(view()->cursor().bottom().pit(),
|
|
||||||
view()->cursor().bottom().pos()));
|
|
||||||
}
|
|
||||||
|
|
||||||
// this is the place where we leave the frontend.
|
// this is the place where we leave the frontend.
|
||||||
// it is the only point at which we start quitting.
|
// it is the only point at which we start quitting.
|
||||||
saveGeometry();
|
saveGeometry();
|
||||||
@ -501,7 +522,9 @@ void GuiView::setGeometry(unsigned int width,
|
|||||||
(void)geometryArg;
|
(void)geometryArg;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
d.setBackground();
|
||||||
|
|
||||||
show();
|
show();
|
||||||
|
|
||||||
// For an unknown reason, the Window title update is not effective for
|
// For an unknown reason, the Window title update is not effective for
|
||||||
@ -610,56 +633,6 @@ void GuiView::update_view_state_qt()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void GuiView::initTab(QWidget* workarea)
|
|
||||||
{
|
|
||||||
// construct the TabWidget with 'false' to have the tabbar at the bottom
|
|
||||||
d.tabWidget = new TabWidget(workarea, true);
|
|
||||||
setCentralWidget(d.tabWidget);
|
|
||||||
QObject::connect(d.tabWidget->tabbar, SIGNAL(currentChanged(int)),
|
|
||||||
this, SLOT(currentTabChanged(int)));
|
|
||||||
QObject::connect(d.tabWidget->closeTabButton, SIGNAL(clicked()),
|
|
||||||
this, SLOT(closeCurrentTab()));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void GuiView::updateTab()
|
|
||||||
{
|
|
||||||
std::vector<string> const & names = theBufferList().getFileNames();
|
|
||||||
|
|
||||||
string cur_title;
|
|
||||||
if (view()->buffer()) {
|
|
||||||
cur_title = view()->buffer()->fileName();
|
|
||||||
}
|
|
||||||
|
|
||||||
// avoid unnecessary tabbar rebuild:
|
|
||||||
// check if something has changed
|
|
||||||
if (d.tabnames == names && d.cur_title == cur_title)
|
|
||||||
return;
|
|
||||||
d.tabnames = names;
|
|
||||||
d.cur_title = cur_title;
|
|
||||||
|
|
||||||
QTabBar & tabbar = *d.tabWidget->tabbar;
|
|
||||||
|
|
||||||
// update when all is done
|
|
||||||
tabbar.blockSignals(true);
|
|
||||||
|
|
||||||
// remove all tab bars
|
|
||||||
d.tabWidget->clearTabbar();
|
|
||||||
|
|
||||||
// rebuild tabbar and function map from scratch
|
|
||||||
if (names.size() > 1) {
|
|
||||||
for(size_t i = 0; i < names.size(); i++) {
|
|
||||||
tabbar.addTab(toqstr(makeDisplayPath(names[i], 30)));
|
|
||||||
// set current tab
|
|
||||||
if (names[i] == cur_title)
|
|
||||||
tabbar.setCurrentIndex(i);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
tabbar.blockSignals(false);
|
|
||||||
d.tabWidget->hideTabsIfNecessary();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void GuiView::closeCurrentTab()
|
void GuiView::closeCurrentTab()
|
||||||
{
|
{
|
||||||
dispatch(FuncRequest(LFUN_BUFFER_CLOSE));
|
dispatch(FuncRequest(LFUN_BUFFER_CLOSE));
|
||||||
@ -668,8 +641,32 @@ void GuiView::closeCurrentTab()
|
|||||||
|
|
||||||
void GuiView::currentTabChanged(int i)
|
void GuiView::currentTabChanged(int i)
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(i >= 0 && size_type(i) < d.tabnames.size());
|
disconnectBuffer();
|
||||||
dispatch(FuncRequest(LFUN_BUFFER_SWITCH, d.tabnames[i]));
|
disconnectBufferView();
|
||||||
|
GuiWorkArea * wa = dynamic_cast<GuiWorkArea *>(d.tab_widget_->widget(i));
|
||||||
|
BOOST_ASSERT(wa);
|
||||||
|
BufferView & bv = wa->bufferView();
|
||||||
|
connectBufferView(bv);
|
||||||
|
connectBuffer(*bv.buffer());
|
||||||
|
bv.updateMetrics(false);
|
||||||
|
bv.cursor().fixIfBroken();
|
||||||
|
wa->setUpdatesEnabled(true);
|
||||||
|
wa->redraw();
|
||||||
|
wa->setFocus();
|
||||||
|
|
||||||
|
updateToc();
|
||||||
|
// Buffer-dependent dialogs should be updated or
|
||||||
|
// hidden. This should go here because some dialogs (eg ToC)
|
||||||
|
// require bv_->text.
|
||||||
|
getDialogs().updateBufferDependent(true);
|
||||||
|
updateMenubar();
|
||||||
|
updateToolbars();
|
||||||
|
updateLayoutChoice();
|
||||||
|
updateWindowTitle();
|
||||||
|
updateStatusBar();
|
||||||
|
|
||||||
|
lyxerr << "currentTabChanged " << i
|
||||||
|
<< "File" << wa->bufferView().buffer()->fileName() << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -741,11 +738,18 @@ bool GuiView::event(QEvent * e)
|
|||||||
|
|
||||||
case QEvent::ShortcutOverride: {
|
case QEvent::ShortcutOverride: {
|
||||||
QKeyEvent * ke = static_cast<QKeyEvent*>(e);
|
QKeyEvent * ke = static_cast<QKeyEvent*>(e);
|
||||||
|
if (d.tab_widget_->count() == 0) {
|
||||||
|
theLyXFunc().setLyXView(this);
|
||||||
|
boost::shared_ptr<QKeySymbol> sym(new QKeySymbol);
|
||||||
|
sym->set(ke);
|
||||||
|
theLyXFunc().processKeySym(sym, q_key_state(ke->modifiers()));
|
||||||
|
e->accept();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (ke->key() == Qt::Key_Tab || ke->key() == Qt::Key_Backtab) {
|
if (ke->key() == Qt::Key_Tab || ke->key() == Qt::Key_Backtab) {
|
||||||
boost::shared_ptr<QKeySymbol> sym(new QKeySymbol);
|
boost::shared_ptr<QKeySymbol> sym(new QKeySymbol);
|
||||||
sym->set(ke);
|
sym->set(ke);
|
||||||
BOOST_ASSERT(work_area_);
|
currentWorkArea()->processKeySym(sym, key_modifier::none);
|
||||||
work_area_->processKeySym(sym, key_modifier::none);
|
|
||||||
e->accept();
|
e->accept();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -773,17 +777,20 @@ void GuiView::show()
|
|||||||
|
|
||||||
void GuiView::busy(bool yes)
|
void GuiView::busy(bool yes)
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(work_area_);
|
if (d.tab_widget_->count()) {
|
||||||
static_cast<GuiWorkArea *>(work_area_)->setUpdatesEnabled(!yes);
|
GuiWorkArea * wa = dynamic_cast<GuiWorkArea *>(d.tab_widget_->currentWidget());
|
||||||
|
BOOST_ASSERT(wa);
|
||||||
|
wa->setUpdatesEnabled(!yes);
|
||||||
|
if (yes)
|
||||||
|
wa->stopBlinkingCursor();
|
||||||
|
else
|
||||||
|
wa->startBlinkingCursor();
|
||||||
|
}
|
||||||
|
|
||||||
if (yes) {
|
if (yes)
|
||||||
work_area_->stopBlinkingCursor();
|
|
||||||
QApplication::setOverrideCursor(Qt::WaitCursor);
|
QApplication::setOverrideCursor(Qt::WaitCursor);
|
||||||
}
|
else
|
||||||
else {
|
|
||||||
work_area_->startBlinkingCursor();
|
|
||||||
QApplication::restoreOverrideCursor();
|
QApplication::restoreOverrideCursor();
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -833,6 +840,96 @@ Toolbars::ToolbarPtr GuiView::makeToolbar(ToolbarInfo const & tbinfo, bool newli
|
|||||||
return Toolbars::ToolbarPtr(Tb);
|
return Toolbars::ToolbarPtr(Tb);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WorkArea * GuiView::workArea(Buffer & buffer)
|
||||||
|
{
|
||||||
|
for (int i = 0; i != d.tab_widget_->count(); ++i) {
|
||||||
|
GuiWorkArea * wa = dynamic_cast<GuiWorkArea *>(d.tab_widget_->widget(i));
|
||||||
|
BOOST_ASSERT(wa);
|
||||||
|
if (wa->bufferView().buffer() == &buffer)
|
||||||
|
return wa;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WorkArea * GuiView::addWorkArea(Buffer & buffer)
|
||||||
|
{
|
||||||
|
GuiWorkArea * wa = new GuiWorkArea(buffer, *this);
|
||||||
|
d.tab_widget_->addTab(wa, toqstr(makeDisplayPath(buffer.fileName(), 30)));
|
||||||
|
wa->bufferView().updateMetrics(false);
|
||||||
|
if (d.stack_widget_)
|
||||||
|
d.stack_widget_->setCurrentWidget(d.tab_widget_);
|
||||||
|
return wa;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WorkArea * GuiView::currentWorkArea()
|
||||||
|
{
|
||||||
|
if (d.tab_widget_->count() == 0)
|
||||||
|
return 0;
|
||||||
|
BOOST_ASSERT(dynamic_cast<GuiWorkArea *>(d.tab_widget_->currentWidget()));
|
||||||
|
return dynamic_cast<GuiWorkArea *>(d.tab_widget_->currentWidget());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
WorkArea const * GuiView::currentWorkArea() const
|
||||||
|
{
|
||||||
|
if (d.tab_widget_->count() == 0)
|
||||||
|
return 0;
|
||||||
|
BOOST_ASSERT(dynamic_cast<GuiWorkArea const *>(d.tab_widget_->currentWidget()));
|
||||||
|
return dynamic_cast<GuiWorkArea const *>(d.tab_widget_->currentWidget());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void GuiView::setCurrentWorkArea(WorkArea * work_area)
|
||||||
|
{
|
||||||
|
BOOST_ASSERT(work_area);
|
||||||
|
|
||||||
|
// Changing work area can result from opening a file so
|
||||||
|
// update the toc in any case.
|
||||||
|
updateToc();
|
||||||
|
|
||||||
|
GuiWorkArea * wa = dynamic_cast<GuiWorkArea *>(work_area);
|
||||||
|
BOOST_ASSERT(wa);
|
||||||
|
d.tab_widget_->setCurrentWidget(wa);
|
||||||
|
wa->setFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void GuiView::removeWorkArea(WorkArea * work_area)
|
||||||
|
{
|
||||||
|
BOOST_ASSERT(work_area);
|
||||||
|
if (work_area == currentWorkArea()) {
|
||||||
|
disconnectBuffer();
|
||||||
|
disconnectBufferView();
|
||||||
|
}
|
||||||
|
|
||||||
|
// removing a work area often results from closing a file so
|
||||||
|
// update the toc in any case.
|
||||||
|
updateToc();
|
||||||
|
|
||||||
|
GuiWorkArea * gwa = dynamic_cast<GuiWorkArea *>(work_area);
|
||||||
|
BOOST_ASSERT(gwa);
|
||||||
|
int index = d.tab_widget_->indexOf(gwa);
|
||||||
|
d.tab_widget_->removeTab(index);
|
||||||
|
|
||||||
|
delete gwa;
|
||||||
|
|
||||||
|
if (d.tab_widget_->count()) {
|
||||||
|
// make sure the next work area is enabled.
|
||||||
|
d.tab_widget_->currentWidget()->setUpdatesEnabled(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDialogs().hideBufferDependent();
|
||||||
|
if (d.stack_widget_) {
|
||||||
|
// No more work area, switch to the background widget.
|
||||||
|
d.setBackground();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
} // namespace frontend
|
} // namespace frontend
|
||||||
} // namespace lyx
|
} // namespace lyx
|
||||||
|
|
||||||
|
@ -75,8 +75,6 @@ public:
|
|||||||
virtual void clearMessage();
|
virtual void clearMessage();
|
||||||
virtual bool hasFocus() const;
|
virtual bool hasFocus() const;
|
||||||
|
|
||||||
virtual void updateTab();
|
|
||||||
|
|
||||||
/// show - display the top-level window
|
/// show - display the top-level window
|
||||||
void show();
|
void show();
|
||||||
|
|
||||||
@ -86,8 +84,6 @@ public:
|
|||||||
/// menu item has been selected
|
/// menu item has been selected
|
||||||
void activated(FuncRequest const &);
|
void activated(FuncRequest const &);
|
||||||
|
|
||||||
void initTab(QWidget* workArea);
|
|
||||||
|
|
||||||
QMenu* createPopupMenu();
|
QMenu* createPopupMenu();
|
||||||
|
|
||||||
Q_SIGNALS:
|
Q_SIGNALS:
|
||||||
@ -118,6 +114,19 @@ protected:
|
|||||||
///
|
///
|
||||||
virtual void moveEvent(QMoveEvent * e);
|
virtual void moveEvent(QMoveEvent * e);
|
||||||
|
|
||||||
|
/// \return the \c Workarea associated to \p Buffer
|
||||||
|
/// \retval 0 if no \c WorkArea is found.
|
||||||
|
WorkArea * workArea(Buffer & buffer);
|
||||||
|
|
||||||
|
/// Add a \c WorkArea
|
||||||
|
/// \return the \c Workarea associated to \p Buffer
|
||||||
|
/// \retval 0 if no \c WorkArea is found.
|
||||||
|
WorkArea * addWorkArea(Buffer & buffer);
|
||||||
|
void setCurrentWorkArea(WorkArea * work_area);
|
||||||
|
void removeWorkArea(WorkArea * work_area);
|
||||||
|
WorkArea const * currentWorkArea() const;
|
||||||
|
WorkArea * currentWorkArea();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
///
|
///
|
||||||
void dragEnterEvent(QDragEnterEvent * ev);
|
void dragEnterEvent(QDragEnterEvent * ev);
|
||||||
|
@ -173,8 +173,8 @@ SyntheticMouseEvent::SyntheticMouseEvent()
|
|||||||
{}
|
{}
|
||||||
|
|
||||||
|
|
||||||
GuiWorkArea::GuiWorkArea(int w, int h, int id, LyXView & lyx_view)
|
GuiWorkArea::GuiWorkArea(Buffer & buf, LyXView & lv)
|
||||||
: WorkArea(id, lyx_view), need_resize_(false), schedule_redraw_(false),
|
: WorkArea(buf, lv), need_resize_(false), schedule_redraw_(false),
|
||||||
preedit_lines_(1)
|
preedit_lines_(1)
|
||||||
{
|
{
|
||||||
screen_ = QPixmap(viewport()->width(), viewport()->height());
|
screen_ = QPixmap(viewport()->width(), viewport()->height());
|
||||||
@ -196,8 +196,6 @@ GuiWorkArea::GuiWorkArea(int w, int h, int id, LyXView & lyx_view)
|
|||||||
|
|
||||||
viewport()->setCursor(Qt::IBeamCursor);
|
viewport()->setCursor(Qt::IBeamCursor);
|
||||||
|
|
||||||
resize(w, h);
|
|
||||||
|
|
||||||
synthetic_mouse_event_.timeout.timeout.connect(
|
synthetic_mouse_event_.timeout.timeout.connect(
|
||||||
boost::bind(&GuiWorkArea::generateSyntheticMouseEvent,
|
boost::bind(&GuiWorkArea::generateSyntheticMouseEvent,
|
||||||
this));
|
this));
|
||||||
@ -258,17 +256,13 @@ void GuiWorkArea::focusInEvent(QFocusEvent * /*event*/)
|
|||||||
// if (theApp() == 0 || &lyx_view_ == theApp()->currentView())
|
// if (theApp() == 0 || &lyx_view_ == theApp()->currentView())
|
||||||
// return;
|
// return;
|
||||||
|
|
||||||
theApp()->setCurrentView(lyx_view_);
|
theApp()->setCurrentView(*lyx_view_);
|
||||||
|
|
||||||
// Repaint the whole screen.
|
// Repaint the whole screen.
|
||||||
// Note: this is different from redraw() as only the backing pixmap
|
// Note: this is different from redraw() as only the backing pixmap
|
||||||
// will be redrawn, which is cheap.
|
// will be redrawn, which is cheap.
|
||||||
viewport()->repaint();
|
viewport()->repaint();
|
||||||
|
|
||||||
// FIXME: it would be better to send a signal "newBuffer()"
|
|
||||||
// in BufferList that could be connected to the different tabbars.
|
|
||||||
lyx_view_.updateTab();
|
|
||||||
|
|
||||||
startBlinkingCursor();
|
startBlinkingCursor();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -459,49 +453,6 @@ void GuiWorkArea::update(int x, int y, int w, int h)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void GuiWorkArea::doGreyOut(QLPainter & pain)
|
|
||||||
{
|
|
||||||
pain.fillRectangle(0, 0, width(), height(),
|
|
||||||
Color::bottomarea);
|
|
||||||
|
|
||||||
//if (!lyxrc.show_banner)
|
|
||||||
// return;
|
|
||||||
LYXERR(Debug::GUI) << "show banner: " << lyxrc.show_banner << endl;
|
|
||||||
/// The text to be written on top of the pixmap
|
|
||||||
QString const text = lyx_version ? QString(lyx_version) : qt_("unknown version");
|
|
||||||
FileName const file = support::libFileSearch("images", "banner", "png");
|
|
||||||
if (file.empty())
|
|
||||||
return;
|
|
||||||
|
|
||||||
QPixmap pm(toqstr(file.absFilename()));
|
|
||||||
if (!pm) {
|
|
||||||
lyxerr << "could not load splash screen: '" << file << "'" << endl;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
QFont font;
|
|
||||||
// The font used to display the version info
|
|
||||||
font.setStyleHint(QFont::SansSerif);
|
|
||||||
font.setWeight(QFont::Bold);
|
|
||||||
font.setPointSize(convert<int>(lyxrc.font_sizes[Font::SIZE_LARGE]));
|
|
||||||
|
|
||||||
int const w = pm.width();
|
|
||||||
int const h = pm.height();
|
|
||||||
|
|
||||||
int x = (width() - w) / 2;
|
|
||||||
int y = (height() - h) / 2;
|
|
||||||
|
|
||||||
pain.drawPixmap(x, y, pm);
|
|
||||||
|
|
||||||
x += 260;
|
|
||||||
y += 270;
|
|
||||||
|
|
||||||
pain.setPen(QColor(255, 255, 0));
|
|
||||||
pain.setFont(font);
|
|
||||||
pain.drawText(x, y, text);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void GuiWorkArea::paintEvent(QPaintEvent * ev)
|
void GuiWorkArea::paintEvent(QPaintEvent * ev)
|
||||||
{
|
{
|
||||||
QRect const rc = ev->rect();
|
QRect const rc = ev->rect();
|
||||||
@ -538,14 +489,6 @@ void GuiWorkArea::expose(int x, int y, int w, int h)
|
|||||||
void GuiWorkArea::updateScreen()
|
void GuiWorkArea::updateScreen()
|
||||||
{
|
{
|
||||||
QLPainter pain(&screen_);
|
QLPainter pain(&screen_);
|
||||||
|
|
||||||
if (greyed_out_) {
|
|
||||||
LYXERR(Debug::GUI) << "splash screen requested" << endl;
|
|
||||||
verticalScrollBar()->hide();
|
|
||||||
doGreyOut(pain);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
verticalScrollBar()->show();
|
verticalScrollBar()->show();
|
||||||
paintText(*buffer_view_, pain);
|
paintText(*buffer_view_, pain);
|
||||||
}
|
}
|
||||||
@ -554,11 +497,9 @@ void GuiWorkArea::updateScreen()
|
|||||||
void GuiWorkArea::showCursor(int x, int y, int h, CursorShape shape)
|
void GuiWorkArea::showCursor(int x, int y, int h, CursorShape shape)
|
||||||
{
|
{
|
||||||
if (schedule_redraw_) {
|
if (schedule_redraw_) {
|
||||||
if (buffer_view_ && buffer_view_->buffer()) {
|
buffer_view_->update(Update::Force);
|
||||||
buffer_view_->update(Update::Force);
|
updateScreen();
|
||||||
updateScreen();
|
viewport()->update(QRect(0, 0, viewport()->width(), viewport()->height()));
|
||||||
viewport()->update(QRect(0, 0, viewport()->width(), viewport()->height()));
|
|
||||||
}
|
|
||||||
schedule_redraw_ = false;
|
schedule_redraw_ = false;
|
||||||
// Show the cursor immediately after the update.
|
// Show the cursor immediately after the update.
|
||||||
hideCursor();
|
hideCursor();
|
||||||
@ -586,11 +527,6 @@ void GuiWorkArea::inputMethodEvent(QInputMethodEvent * e)
|
|||||||
docstring const & preedit_string
|
docstring const & preedit_string
|
||||||
= qstring_to_ucs4(e->preeditString());
|
= qstring_to_ucs4(e->preeditString());
|
||||||
|
|
||||||
if(greyed_out_) {
|
|
||||||
e->ignore();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!commit_string.isEmpty()) {
|
if (!commit_string.isEmpty()) {
|
||||||
|
|
||||||
LYXERR(Debug::KEY) << BOOST_CURRENT_FUNCTION
|
LYXERR(Debug::KEY) << BOOST_CURRENT_FUNCTION
|
||||||
|
@ -88,10 +88,11 @@ class GuiWorkArea : public QAbstractScrollArea, public WorkArea
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
///
|
///
|
||||||
GuiWorkArea(int width, int height, int id, LyXView & lyx_view);
|
GuiWorkArea(Buffer & buffer, LyXView & lv);
|
||||||
|
|
||||||
///
|
///
|
||||||
bool hasFocus() const { return QAbstractScrollArea::hasFocus(); }
|
bool hasFocus() const { return QAbstractScrollArea::hasFocus(); }
|
||||||
|
bool isVisible() const { return QAbstractScrollArea::isVisible(); }
|
||||||
|
|
||||||
/// return the width of the content pane
|
/// return the width of the content pane
|
||||||
virtual int width() const { return viewport()->width(); }
|
virtual int width() const { return viewport()->width(); }
|
||||||
|
@ -386,37 +386,34 @@ Buffer * getChildBuffer(Buffer const & buffer, InsetCommandParams const & params
|
|||||||
|
|
||||||
} // namespace anon
|
} // namespace anon
|
||||||
|
|
||||||
|
/// return true if the file is or got loaded.
|
||||||
Buffer * loadIfNeeded(Buffer const & buffer, InsetCommandParams const & params)
|
Buffer * loadIfNeeded(Buffer const & parent, InsetCommandParams const & params)
|
||||||
{
|
{
|
||||||
if (isVerbatim(params) || isListings(params))
|
if (isVerbatim(params) || isListings(params))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
FileName const included_file = includedFilename(buffer, params);
|
string const parent_filename = parent.fileName();
|
||||||
|
FileName const included_file = makeAbsPath(to_utf8(params["filename"]),
|
||||||
|
onlyPath(parent_filename));
|
||||||
|
|
||||||
if (!isLyXFilename(included_file.absFilename()))
|
if (!isLyXFilename(included_file.absFilename()))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
Buffer * buf = theBufferList().getBuffer(included_file.absFilename());
|
Buffer * child = theBufferList().getBuffer(included_file.absFilename());
|
||||||
if (!buf) {
|
if (!child) {
|
||||||
// the readonly flag can/will be wrong, not anymore I think.
|
// the readonly flag can/will be wrong, not anymore I think.
|
||||||
if (!fs::exists(included_file.toFilesystemEncoding()))
|
if (!fs::exists(included_file.toFilesystemEncoding()))
|
||||||
return false;
|
return 0;
|
||||||
if (use_gui) {
|
|
||||||
lyx::dispatch(FuncRequest(LFUN_BUFFER_CHILD_OPEN,
|
child = theBufferList().newBuffer(included_file.absFilename());
|
||||||
included_file.absFilename() + "|true"));
|
if (!loadLyXFile(child, included_file)) {
|
||||||
buf = theBufferList().getBuffer(included_file.absFilename());
|
//close the buffer we just opened
|
||||||
}
|
theBufferList().close(child, false);
|
||||||
else {
|
return 0;
|
||||||
buf = theBufferList().newBuffer(included_file.absFilename());
|
|
||||||
if (!loadLyXFile(buf, included_file)) {
|
|
||||||
//close the buffer we just opened
|
|
||||||
theBufferList().close(buf, false);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
buf->setParentName(parentFilename(buffer));
|
child->setParentName(parent_filename);
|
||||||
return buf;
|
return child;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -768,7 +765,7 @@ InsetInclude::getBibfilesCache(Buffer const & buffer) const
|
|||||||
|
|
||||||
bool InsetInclude::metrics(MetricsInfo & mi, Dimension & dim) const
|
bool InsetInclude::metrics(MetricsInfo & mi, Dimension & dim) const
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(mi.base.bv && mi.base.bv->buffer());
|
BOOST_ASSERT(mi.base.bv);
|
||||||
|
|
||||||
bool use_preview = false;
|
bool use_preview = false;
|
||||||
if (RenderPreview::status() != LyXRC::PREVIEW_OFF) {
|
if (RenderPreview::status() != LyXRC::PREVIEW_OFF) {
|
||||||
@ -801,7 +798,7 @@ void InsetInclude::draw(PainterInfo & pi, int x, int y) const
|
|||||||
{
|
{
|
||||||
setPosCache(pi, x, y);
|
setPosCache(pi, x, y);
|
||||||
|
|
||||||
BOOST_ASSERT(pi.base.bv && pi.base.bv->buffer());
|
BOOST_ASSERT(pi.base.bv);
|
||||||
|
|
||||||
bool use_preview = false;
|
bool use_preview = false;
|
||||||
if (RenderPreview::status() != LyXRC::PREVIEW_OFF) {
|
if (RenderPreview::status() != LyXRC::PREVIEW_OFF) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user