lyx_mirror/src/frontends/qt4/GuiErrorList.cpp
Enrico Forestieri 13dd04602a Generalize check to generic non-editable insets.
git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@38257 a592a061-630c-0410-9148-cb99ea01b6c8
2011-04-05 15:40:00 +00:00

212 lines
4.7 KiB
C++

/**
* \file GuiErrorList.cpp
* This file is part of LyX, the document processor.
* Licence details can be found in the file COPYING.
*
* \author Alfredo Braunstein
*
* Full author contact details are available in file CREDITS.
*/
#include <config.h>
#include "GuiErrorList.h"
#include "GuiView.h"
#include "qt_helpers.h"
#include "Buffer.h"
#include "BufferView.h"
#include "FuncRequest.h"
#include "BufferList.h"
#include "ParIterator.h"
#include "Text.h"
#include "support/debug.h"
#include "support/gettext.h"
#include "support/lstrings.h"
#include <QListWidget>
#include <QPushButton>
#include <QShowEvent>
#include <QTextBrowser>
using namespace std;
using namespace lyx::support;
namespace {
string const guiErrorType(string const s)
{
if (s == "docbook")
return N_("DocBook");
else if (s == "literate")
return N_("Literate");
else if (s == "platex")
return N_("pLaTeX");
else if (s == "latex")
return N_("LaTeX");
return s;
}
} // namespace anon
namespace lyx {
namespace frontend {
GuiErrorList::GuiErrorList(GuiView & lv)
: GuiDialog(lv, "errorlist", qt_("Error List"))
{
setupUi(this);
connect(closePB, SIGNAL(clicked()),
this, SLOT(slotClose()));
connect(viewLogPB, SIGNAL(clicked()),
this, SLOT(viewLog()));
connect(errorsLW, SIGNAL(currentRowChanged(int)),
this, SLOT(select()));
bc().setPolicy(ButtonPolicy::OkCancelPolicy);
bc().setCancel(closePB);
}
void GuiErrorList::showEvent(QShowEvent * e)
{
select();
paramsToDialog();
e->accept();
}
void GuiErrorList::select()
{
int const item = errorsLW->row(errorsLW->currentItem());
if (item == -1)
return;
goTo(item);
descriptionTB->setPlainText(
toqstr(errorList()[item].description));
}
void GuiErrorList::viewLog()
{
if (&buffer() != buf_) {
if (!theBufferList().isLoaded(buf_))
return;
FuncRequest fr(LFUN_BUFFER_SWITCH, buf_->absFileName());
dispatch(fr);
}
dispatch(FuncRequest(LFUN_DIALOG_SHOW, "latexlog"));
}
void GuiErrorList::paramsToDialog()
{
setTitle(toqstr(name_));
errorsLW->clear();
descriptionTB->setPlainText(QString());
ErrorList const & el = errorList();
ErrorList::const_iterator it = el.begin();
ErrorList::const_iterator const en = el.end();
for (; it != en; ++it)
errorsLW->addItem(toqstr(it->error));
errorsLW->setCurrentRow(0);
}
ErrorList const & GuiErrorList::errorList() const
{
Buffer const * buffer = from_master_
? bufferview()->buffer().masterBuffer()
: &bufferview()->buffer();
if (buffer == buf_)
error_list_ = buffer->errorList(error_type_);
return error_list_;
}
bool GuiErrorList::initialiseParams(string const & data)
{
from_master_ = prefixIs(data, "from_master|");
string error_type = data;
if (from_master_)
error_type = split(data, '|');
error_type_ = error_type;
buf_ = from_master_ ?
bufferview()->buffer().masterBuffer()
: &bufferview()->buffer();
name_ = bformat(_("%1$s Errors (%2$s)"),
_(guiErrorType(error_type)),
from_utf8(buf_->absFileName()));
paramsToDialog();
return true;
}
bool GuiErrorList::goTo(int item)
{
ErrorItem const & err = errorList()[item];
if (err.par_id == -1)
return false;
Buffer const * errbuf = err.buffer ? err.buffer : buf_;
if (&buffer() != errbuf) {
if (!theBufferList().isLoaded(errbuf))
return false;
FuncRequest fr(LFUN_BUFFER_SWITCH, errbuf->absFileName());
dispatch(fr);
}
DocIterator dit = errbuf->getParFromID(err.par_id);
if (dit == doc_iterator_end(errbuf)) {
// FIXME: Happens when loading a read-only doc with
// unknown layout. Should this be the case?
LYXERR0("par id " << err.par_id << " not found");
return false;
}
// Don't try to highlight the content of non-editable insets
while (!dit.inset().editable())
dit.backwardPos();
// Now make the selection.
BufferView * bv = const_cast<BufferView *>(bufferview());
if (bv->selectIfEmpty(dit)) {
// The paragraph is empty but can be selected
bv->processUpdateFlags(Update::Force | Update::FitCursor);
return true;
}
if (dit.empty()) {
// The paragraph is empty and cannot be selected
return false;
}
// if pos_end is 0, this means it is end-of-paragraph
pos_type const s = dit.paragraph().size();
pos_type const end = err.pos_end ? min(err.pos_end, s) : s;
pos_type const start = min(err.pos_start, end);
pos_type const range = end == start ? s - start : end - start;
// end-of-paragraph cannot be highlighted, so highlight the last thing
dit.pos() = range ? start : end - 1;
// FIXME LFUN
// If we used an LFUN, we would not need these lines:
bv->putSelectionAt(dit, max(range, pos_type(1)), false);
bv->processUpdateFlags(Update::Force | Update::FitCursor);
return true;
}
Dialog * createGuiErrorList(GuiView & lv) { return new GuiErrorList(lv); }
} // namespace frontend
} // namespace lyx
#include "moc_GuiErrorList.cpp"