lyx_mirror/src/frontends/qt/GuiLog.cpp
Enrico Forestieri 635a7d77dd Allow compiling with Qt6
This commit allows compiling LyX with Qt6 when using autotools.
For a successful compilation the following 2 conditions must be met.

1) The Qt6 qmake has to come first in PATH, so that the command
   "qmake -v | grep -o 'Qt version .'" returns "Qt version 6".
2) The --enable-qt6 switch has to be passed to the configure command.

If --enable-qt6 is used but Qt6 is not found, Qt5 is tried as a fallback.
If also Qt5 is not found, configuring for Qt4 is attempted.
If --enable-qt6 is not used, then things go as usual. This means that Qt5
is tried first and then Qt4, unless --disable-qt5 is used, in which case
Qt4 is directly attempted. This means that existing scripts should
continue working unmodified.

LyX should compile with Qt6 on windows and linux, and possibly also on
mac, but I could not test that. However, it is not guaranteed that it
works as it should. In particular I am not sure that I got right the
conversion from QRegExp to QRegularExpression. For sure, the syntax
highlighting seems to not work right. Someone in the know should take
a look at that. I am able to load documents and compile them but some
thourough testing is needed. However, when compiling for Qt5 or Qt4,
I tried to make sure that the functionality is preserved.
2021-03-15 17:09:09 +01:00

378 lines
8.7 KiB
C++

/**
* \file GuiLog.cpp
* This file is part of LyX, the document processor.
* Licence details can be found in the file COPYING.
*
* \author John Levon
* \author Angus Leeming
* \author Jürgen Spitzmüller
*
* Full author contact details are available in file CREDITS.
*/
#include <config.h>
#include "GuiLog.h"
#include "GuiApplication.h"
#include "qt_helpers.h"
#include "Lexer.h"
#include "frontends/Clipboard.h"
#include "support/docstring.h"
#include "support/gettext.h"
#include "support/lstrings.h"
#include <QTextBrowser>
#include <QSyntaxHighlighter>
#include <QClipboard>
#include <fstream>
#include <sstream>
using namespace std;
using namespace lyx::support;
namespace lyx {
namespace frontend {
// Regular expressions needed at several places
// FIXME: These regexes are incomplete. It would be good if we could collect those used in LaTeX::scanLogFile
// and LaTeX::scanBlgFile and re-use them here!(spitz, 2013-05-27)
#if QT_VERSION < 0x060000
// Information
QRegExp exprInfo("^(Document Class:|LaTeX Font Info:|File:|Package:|Language:|.*> INFO - |\\(|\\\\).*$");
// Warnings
QRegExp exprWarning("^(## Warning|LaTeX Warning|LaTeX Font Warning|Package [\\w\\.]+ Warning|Class \\w+ Warning|Warning--|Underfull|Overfull|.*> WARN - ).*$");
// Errors
QRegExp exprError("^(ERROR: |!|.*---line [0-9]+ of file|.*> FATAL - |.*> ERROR - |Missing character: There is no ).*$");
#else
// Information
QRegularExpression exprInfo("^(Document Class:|LaTeX Font Info:|File:|Package:|Language:|.*> INFO - |\\(|\\\\).*$");
// Warnings
QRegularExpression exprWarning("^(## Warning|LaTeX Warning|LaTeX Font Warning|Package [\\w\\.]+ Warning|Class \\w+ Warning|Warning--|Underfull|Overfull|.*> WARN - ).*$");
// Errors
QRegularExpression exprError("^(ERROR: |!|.*---line [0-9]+ of file|.*> FATAL - |.*> ERROR - |Missing character: There is no ).*$");
#endif
/////////////////////////////////////////////////////////////////////
//
// LogHighlighter
//
////////////////////////////////////////////////////////////////////
class LogHighlighter : public QSyntaxHighlighter
{
public:
LogHighlighter(QTextDocument * parent);
private:
void highlightBlock(QString const & text) override;
private:
QTextCharFormat infoFormat;
QTextCharFormat warningFormat;
QTextCharFormat errorFormat;
};
LogHighlighter::LogHighlighter(QTextDocument * parent)
: QSyntaxHighlighter(parent)
{
infoFormat.setForeground(Qt::darkGray);
warningFormat.setForeground(Qt::darkBlue);
errorFormat.setForeground(Qt::red);
}
void LogHighlighter::highlightBlock(QString const & text)
{
#if QT_VERSION < 0x060000
// Info
int index = exprInfo.indexIn(text);
while (index >= 0) {
int length = exprInfo.matchedLength();
setFormat(index, length, infoFormat);
index = exprInfo.indexIn(text, index + length);
}
// LaTeX Warning:
index = exprWarning.indexIn(text);
while (index >= 0) {
int length = exprWarning.matchedLength();
setFormat(index, length, warningFormat);
index = exprWarning.indexIn(text, index + length);
}
// ! error
index = exprError.indexIn(text);
while (index >= 0) {
int length = exprError.matchedLength();
setFormat(index, length, errorFormat);
index = exprError.indexIn(text, index + length);
}
#else
// Info
QRegularExpressionMatch match = exprInfo.match(text);
int index = match.capturedStart(1);
while (index >= 0) {
int length = match.capturedEnd(1) - index;
setFormat(index, length, infoFormat);
match = exprInfo.match(text, index + length);
index = match.capturedStart(1);
}
// LaTeX Warning:
match = exprWarning.match(text);
index = match.capturedStart(1);
while (index >= 0) {
int length = match.capturedEnd(1) - index;
setFormat(index, length, warningFormat);
match = exprWarning.match(text, index + length);
index = match.capturedStart(1);
}
// ! error
match = exprError.match(text);
index = match.capturedStart(1);
while (index >= 0) {
int length = match.capturedEnd(1) - index;
setFormat(index, length, errorFormat);
match = exprError.match(text, index + length);
index = match.capturedStart(1);
}
#endif
}
/////////////////////////////////////////////////////////////////////
//
// GuiLog
//
/////////////////////////////////////////////////////////////////////
GuiLog::GuiLog(GuiView & lv)
: GuiDialog(lv, "log", qt_("LaTeX Log")), type_(LatexLog)
{
setupUi(this);
connect(buttonBox, SIGNAL(clicked(QAbstractButton *)),
this, SLOT(slotButtonBox(QAbstractButton *)));
connect(updatePB, SIGNAL(clicked()), this, SLOT(updateContents()));
connect(findPB, SIGNAL(clicked()), this, SLOT(find()));
connect(findLE, SIGNAL(returnPressed()), this, SLOT(find()));
connect(logTypeCO, SIGNAL(activated(int)),
this, SLOT(typeChanged(int)));
bc().setPolicy(ButtonPolicy::OkCancelPolicy);
// set syntax highlighting
highlighter = new LogHighlighter(logTB->document());
logTB->setReadOnly(true);
logTB->setFont(guiApp->typewriterSystemFont());
QPushButton * closePB = buttonBox->button(QDialogButtonBox::Close);
closePB->setAutoDefault(false);
}
void GuiLog::updateContents()
{
setTitle(toqstr(title()));
ostringstream ss;
getContents(ss);
logTB->setPlainText(toqstr(ss.str()));
nextErrorPB->setEnabled(contains(exprError));
nextWarningPB->setEnabled(contains(exprWarning));
}
void GuiLog::typeChanged(int i)
{
string const type =
fromqstr(logTypeCO->itemData(i).toString());
string ext;
if (type == "latex")
ext = "log";
else if (type == "bibtex")
ext = "blg";
else if (type == "index")
ext = "ilg";
if (!ext.empty())
logfile_.changeExtension(ext);
updateContents();
}
void GuiLog::find()
{
logTB->find(findLE->text());
}
void GuiLog::on_nextErrorPB_clicked()
{
goTo(exprError);
}
void GuiLog::on_nextWarningPB_clicked()
{
goTo(exprWarning);
}
void GuiLog::on_openDirPB_clicked()
{
showDirectory(logfile_.onlyPath());
}
#if QT_VERSION < 0x060000
void GuiLog::goTo(QRegExp const & exp) const
#else
void GuiLog::goTo(QRegularExpression const & exp) const
#endif
{
QTextCursor const newc =
logTB->document()->find(exp, logTB->textCursor());
logTB->setTextCursor(newc);
}
#if QT_VERSION < 0x060000
bool GuiLog::contains(QRegExp const & exp) const
#else
bool GuiLog::contains(QRegularExpression const & exp) const
#endif
{
return !logTB->document()->find(exp, logTB->textCursor()).isNull();
}
bool GuiLog::initialiseParams(string const & sdata)
{
istringstream is(sdata);
Lexer lex;
lex.setStream(is);
string logtype, logfile;
lex >> logtype;
if (lex) {
lex.next(true);
logfile = lex.getString();
}
if (!lex)
// Parsing of the data failed.
return false;
logTypeCO->setEnabled(logtype == "latex");
logTypeCO->clear();
FileName log(logfile);
if (logtype == "latex") {
type_ = LatexLog;
logTypeCO->addItem(qt_("LaTeX"), toqstr(logtype));
FileName tmp = log;
tmp.changeExtension("blg");
if (tmp.exists()) {
if (support::contains(tmp.fileContents("UTF-8"), from_ascii("This is Biber")))
logTypeCO->addItem(qt_("Biber"), QString("bibtex"));
else
logTypeCO->addItem(qt_("BibTeX"), QString("bibtex"));
}
tmp.changeExtension("ilg");
if (tmp.exists())
logTypeCO->addItem(qt_("Index"), QString("index"));
// FIXME: not sure "literate" still works.
} else if (logtype == "literate") {
type_ = LiterateLog;
logTypeCO->addItem(qt_("Literate"), toqstr(logtype));
} else if (logtype == "lyx2lyx") {
type_ = Lyx2lyxLog;
logTypeCO->addItem(qt_("LyX2LyX"), toqstr(logtype));
} else if (logtype == "vc") {
type_ = VCLog;
logTypeCO->addItem(qt_("Version Control"), toqstr(logtype));
} else
return false;
logfile_ = log;
updateContents();
return true;
}
void GuiLog::clearParams()
{
logfile_.erase();
}
docstring GuiLog::title() const
{
switch (type_) {
case LatexLog:
return _("LaTeX Log");
case LiterateLog:
return _("Literate Programming Build Log");
case Lyx2lyxLog:
return _("lyx2lyx Error Log");
case VCLog:
return _("Version Control Log");
default:
return docstring();
}
}
void GuiLog::getContents(ostream & ss) const
{
ifstream in(logfile_.toFilesystemEncoding().c_str());
bool success = false;
// FIXME UNICODE
// Our caller interprets the file contents as UTF8, but is that
// correct?
// spitz: No it isn't (generally). The log file encoding depends on the TeX
// _output_ encoding (T1 etc.). We should account for that. See #10728.
if (in) {
ss << in.rdbuf();
success = ss.good();
}
if (success)
return;
switch (type_) {
case LatexLog:
ss << to_utf8(_("Log file not found."));
break;
case LiterateLog:
ss << to_utf8(_("No literate programming build log file found."));
break;
case Lyx2lyxLog:
ss << to_utf8(_("No lyx2lyx error log file found."));
break;
case VCLog:
ss << to_utf8(_("No version control log file found."));
break;
}
}
} // namespace frontend
} // namespace lyx
#include "moc_GuiLog.cpp"