#include <config.h>
+#include "GuiApplication.h"
#include "GuiViewSource.h"
#include "LaTeXHighlighter.h"
#include "qt_helpers.h"
-#include "Application.h"
-#include "BufferView.h"
#include "Buffer.h"
+#include "BufferParams.h"
+#include "BufferView.h"
#include "Cursor.h"
-#include "gettext.h"
+#include "Format.h"
#include "Paragraph.h"
#include "TexRow.h"
+#include "support/debug.h"
+#include "support/lassert.h"
#include "support/docstream.h"
+#include "support/gettext.h"
+
+#include <boost/crc.hpp>
+#include <QSettings>
#include <QTextCursor>
#include <QTextDocument>
-#include <boost/tuple/tuple.hpp>
+#include <QVariant>
-using std::string;
+using namespace std;
namespace lyx {
namespace frontend {
-ViewSourceWidget::ViewSourceWidget(GuiViewSource & controller)
- : controller_(controller), document_(new QTextDocument(this)),
- highlighter_(new LaTeXHighlighter(document_))
+ViewSourceWidget::ViewSourceWidget()
+ : bv_(0), document_(new QTextDocument(this)),
+ highlighter_(new LaTeXHighlighter(document_)),
+ force_getcontent_(true)
{
setupUi(this);
- setWindowTitle(qt_("LaTeX Source"));
- connect(viewFullSourceCB, SIGNAL(clicked()),
- this, SLOT(updateView()));
+ connect(contentsCO, SIGNAL(activated(int)),
+ this, SLOT(contentsChanged()));
connect(autoUpdateCB, SIGNAL(toggled(bool)),
updatePB, SLOT(setDisabled(bool)));
+ connect(autoUpdateCB, SIGNAL(toggled(bool)),
+ this, SLOT(updateView()));
connect(updatePB, SIGNAL(clicked()),
this, SLOT(updateView()));
+ connect(outputFormatCO, SIGNAL(activated(int)),
+ this, SLOT(updateView()));
// setting a document at this point trigger an assertion in Qt
// so we disable the signals here:
viewSourceTV->setReadOnly(true);
///dialog_->viewSourceTV->setAcceptRichText(false);
// this is personal. I think source code should be in fixed-size font
- QFont font(toqstr(theApp()->typewriterFontName()));
+ QFont font(guiApp->typewriterFontName());
font.setKerning(false);
font.setFixedPitch(true);
font.setStyleHint(QFont::TypeWriter);
}
-void ViewSourceWidget::updateView()
+static size_t crcCheck(docstring const & s)
+{
+ boost::crc_32_type crc;
+ crc.process_bytes(&s[0], sizeof(char_type) * s.size());
+ return crc.checksum();
+}
+
+
+/** get the source code of selected paragraphs, or the whole document
+ \param fullSource get full source code
+ \return true if the content has changed since last call.
+ */
+static bool getContent(BufferView const * view, Buffer::OutputWhat output,
+ QString & qstr, string const format, bool force_getcontent)
+{
+ // get the *top* level paragraphs that contain the cursor,
+ // or the selected text
+ pit_type par_begin;
+ pit_type par_end;
+
+ if (!view->cursor().selection()) {
+ par_begin = view->cursor().bottom().pit();
+ par_end = par_begin;
+ } else {
+ par_begin = view->cursor().selectionBegin().bottom().pit();
+ par_end = view->cursor().selectionEnd().bottom().pit();
+ }
+ if (par_begin > par_end)
+ swap(par_begin, par_end);
+ odocstringstream ostr;
+ view->buffer().getSourceCode(ostr, format, par_begin, par_end + 1, output);
+ docstring s = ostr.str();
+ static size_t crc = 0;
+ size_t newcrc = crcCheck(s);
+ if (newcrc == crc && !force_getcontent)
+ return false;
+ crc = newcrc;
+ qstr = toqstr(s);
+ return true;
+}
+
+
+void ViewSourceWidget::setBufferView(BufferView const * bv)
+{
+ if (bv_ != bv)
+ force_getcontent_ = true;
+ bv_ = bv;
+ setEnabled(bv ? true : false);
+}
+
+
+void ViewSourceWidget::contentsChanged()
{
if (autoUpdateCB->isChecked())
- update(viewFullSourceCB->isChecked());
+ updateView();
+}
+
+
+void ViewSourceWidget::updateView()
+{
+ if (!bv_) {
+ document_->setPlainText(QString());
+ setEnabled(false);
+ return;
+ }
+
+ setEnabled(true);
+
+ string const format = fromqstr(outputFormatCO->itemData(
+ outputFormatCO->currentIndex()).toString());
+
+ QString content;
+ Buffer::OutputWhat output = Buffer::CurrentParagraph;
+ if (contentsCO->currentIndex() == 1)
+ output = Buffer::FullSource;
+ else if (contentsCO->currentIndex() == 2)
+ output = Buffer::OnlyPreamble;
+ else if (contentsCO->currentIndex() == 3)
+ output = Buffer::OnlyBody;
+
+ if (getContent(bv_, output, content, format, force_getcontent_))
+ document_->setPlainText(content);
+
+ CursorSlice beg = bv_->cursor().selectionBegin().bottom();
+ CursorSlice end = bv_->cursor().selectionEnd().bottom();
+ int const begrow = bv_->buffer().texrow().
+ getRowFromIdPos(beg.paragraph().id(), beg.pos());
+ int endrow = bv_->buffer().texrow().
+ getRowFromIdPos(end.paragraph().id(), end.pos());
+ int const nextendrow = bv_->buffer().texrow().
+ getRowFromIdPos(end.paragraph().id(), end.pos() + 1);
+ if (endrow != nextendrow)
+ endrow = nextendrow - 1;
- int beg, end;
- boost::tie(beg, end) = controller_.getRows();
QTextCursor c = QTextCursor(viewSourceTV->document());
- c.movePosition(QTextCursor::NextBlock, QTextCursor::MoveAnchor, beg);
+ c.movePosition(QTextCursor::NextBlock, QTextCursor::MoveAnchor, begrow);
c.select(QTextCursor::BlockUnderCursor);
- c.movePosition(QTextCursor::NextBlock, QTextCursor::KeepAnchor, end - beg + 1);
+ c.movePosition(QTextCursor::NextBlock, QTextCursor::KeepAnchor,
+ endrow - begrow + 1);
viewSourceTV->setTextCursor(c);
}
-void ViewSourceWidget::update(bool full_source)
+void ViewSourceWidget::updateDefaultFormat()
{
- document_->setPlainText(controller_.getContent(full_source));
+ if (!bv_)
+ return;
+
+ outputFormatCO->blockSignals(true);
+ outputFormatCO->clear();
+ outputFormatCO->addItem(qt_("Default"),
+ QVariant(QString("default")));
+ typedef vector<Format const *> Formats;
+ Formats formats = bv_->buffer().params().exportableFormats(true);
+ Formats::const_iterator cit = formats.begin();
+ Formats::const_iterator end = formats.end();
+ for (; cit != end; ++cit)
+ outputFormatCO->addItem(qt_((*cit)->prettyname()),
+ QVariant(toqstr((*cit)->name())));
+ outputFormatCO->blockSignals(false);
}
-GuiViewSource::GuiViewSource(GuiView & parent, Qt::DockWidgetArea area, Qt::WindowFlags flags)
- : DockView(parent, "view-source", area, flags)
+GuiViewSource::GuiViewSource(GuiView & parent,
+ Qt::DockWidgetArea area, Qt::WindowFlags flags)
+ : DockView(parent, "view-source", qt_("LaTeX Source"), area, flags)
{
- widget_ = new ViewSourceWidget(*this);
+ widget_ = new ViewSourceWidget();
setWidget(widget_);
- setWindowTitle(widget_->windowTitle());
}
void GuiViewSource::updateView()
{
- widget_->updateView();
-}
-
-
-bool GuiViewSource::initialiseParams(string const & /*source*/)
-{
- setWindowTitle(title());
- return true;
+ if (widget_->autoUpdateCB->isChecked()) {
+ widget_->setBufferView(bufferview());
+ widget_->updateView();
+ }
}
-QString GuiViewSource::getContent(bool fullSource)
+void GuiViewSource::enableView(bool enable)
{
- // get the *top* level paragraphs that contain the cursor,
- // or the selected text
- pit_type par_begin;
- pit_type par_end;
-
- BufferView * view = bufferview();
- if (!view->cursor().selection()) {
- par_begin = view->cursor().bottom().pit();
- par_end = par_begin;
- } else {
- par_begin = view->cursor().selectionBegin().bottom().pit();
- par_end = view->cursor().selectionEnd().bottom().pit();
- }
- if (par_begin > par_end)
- std::swap(par_begin, par_end);
- odocstringstream ostr;
- view->buffer().getSourceCode(ostr, par_begin, par_end + 1, fullSource);
- return toqstr(ostr.str());
+ widget_->setBufferView(bufferview());
+ widget_->updateDefaultFormat();
+ if (!enable)
+ // In the opposite case, updateView() will be called anyway.
+ widget_->updateView();
}
-std::pair<int, int> GuiViewSource::getRows() const
+bool GuiViewSource::initialiseParams(string const & /*source*/)
{
- BufferView const * view = bufferview();
- CursorSlice beg = view->cursor().selectionBegin().bottom();
- CursorSlice end = view->cursor().selectionEnd().bottom();
-
- int begrow = view->buffer().texrow().
- getRowFromIdPos(beg.paragraph().id(), beg.pos());
- int endrow = view->buffer().texrow().
- getRowFromIdPos(end.paragraph().id(), end.pos());
- int nextendrow = view->buffer().texrow().
- getRowFromIdPos(end.paragraph().id(), end.pos() + 1);
- return std::make_pair(begrow, endrow == nextendrow ? endrow : (nextendrow - 1));
+ setWindowTitle(title());
+ return true;
}
case LITERATE:
return qt_("Literate Source");
}
- BOOST_ASSERT(false);
+ LASSERT(false, /**/);
return QString();
}
-Dialog * createGuiViewSource(LyXView & lv)
+void GuiViewSource::saveSession() const
+{
+ Dialog::saveSession();
+ QSettings settings;
+ // see below
+ // settings.setValue(
+ // sessionKey() + "/output", widget_->contentsCO->currentIndex());
+ settings.setValue(
+ sessionKey() + "/autoupdate", widget_->autoUpdateCB->isChecked());
+}
+
+
+void GuiViewSource::restoreSession()
+{
+ DockView::restoreSession();
+ // FIXME: Full source updating is too slow to be done at startup.
+ //widget_->outputCO-setCurrentIndex(
+ // settings.value(sessionKey() + "/output", false).toInt());
+ widget_->contentsCO->setCurrentIndex(0);
+ QSettings settings;
+ widget_->autoUpdateCB->setChecked(
+ settings.value(sessionKey() + "/autoupdate", true).toBool());
+ widget_->updateView();
+}
+
+
+Dialog * createGuiViewSource(GuiView & lv)
{
- return new GuiViewSource(static_cast<GuiView &>(lv));
+ return new GuiViewSource(lv);
}
} // namespace frontend
} // namespace lyx
-#include "GuiViewSource_moc.cpp"
+#include "moc_GuiViewSource.cpp"