2006-03-05 17:24:44 +00:00
|
|
|
/**
|
2007-04-26 03:53:02 +00:00
|
|
|
* \file qt_helpers.cpp
|
2006-03-05 17:24:44 +00:00
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
|
|
|
* \author Dekel Tsur
|
2008-11-14 15:58:50 +00:00
|
|
|
* \author Jürgen Spitzmüller
|
2007-04-05 14:58:15 +00:00
|
|
|
* \author Richard Heck
|
2006-03-05 17:24:44 +00:00
|
|
|
*
|
|
|
|
* Full author contact details are available in file CREDITS.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "qt_helpers.h"
|
|
|
|
|
2008-02-07 00:05:18 +00:00
|
|
|
#include "FileDialog.h"
|
2007-10-24 22:55:02 +00:00
|
|
|
#include "LengthCombo.h"
|
|
|
|
|
2008-02-18 07:14:42 +00:00
|
|
|
#include "frontends/alert.h"
|
|
|
|
|
2008-05-16 13:49:49 +00:00
|
|
|
#include "BufferParams.h"
|
|
|
|
#include "FloatList.h"
|
2007-11-05 22:20:23 +00:00
|
|
|
#include "Language.h"
|
2008-05-16 13:49:49 +00:00
|
|
|
#include "TextClass.h"
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2009-07-12 15:44:26 +00:00
|
|
|
#include "support/convert.h"
|
2008-02-18 07:14:42 +00:00
|
|
|
#include "support/debug.h"
|
|
|
|
#include "support/gettext.h"
|
2020-08-03 14:15:09 +02:00
|
|
|
#include "support/Length.h"
|
2006-03-05 17:24:44 +00:00
|
|
|
#include "support/lstrings.h"
|
2007-11-05 22:20:23 +00:00
|
|
|
#include "support/lyxalgo.h"
|
|
|
|
#include "support/os.h"
|
|
|
|
#include "support/Package.h"
|
2013-02-06 11:36:58 +01:00
|
|
|
#include "support/PathChanger.h"
|
2007-11-05 22:20:23 +00:00
|
|
|
#include "support/Systemcall.h"
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2017-05-15 02:01:58 +02:00
|
|
|
#include <QApplication>
|
2007-04-05 14:58:15 +00:00
|
|
|
#include <QCheckBox>
|
2008-03-08 07:59:47 +00:00
|
|
|
#include <QComboBox>
|
2007-09-15 15:42:22 +00:00
|
|
|
#include <QLineEdit>
|
2009-07-12 15:44:26 +00:00
|
|
|
#include <QLocale>
|
2008-03-08 07:59:47 +00:00
|
|
|
#include <QPalette>
|
|
|
|
#include <QSet>
|
2016-06-05 21:35:35 +02:00
|
|
|
#include <QTextLayout>
|
|
|
|
#include <QTextDocument>
|
|
|
|
#include <QToolTip>
|
2007-11-05 22:20:23 +00:00
|
|
|
|
2006-03-05 17:24:44 +00:00
|
|
|
#include <algorithm>
|
2007-11-05 22:20:23 +00:00
|
|
|
#include <fstream>
|
2007-11-14 02:01:50 +00:00
|
|
|
#include <locale>
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2008-04-20 19:56:01 +00:00
|
|
|
// for FileFilter.
|
|
|
|
// FIXME: Remove
|
2010-06-29 17:09:40 +00:00
|
|
|
#include "support/regex.h"
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
|
2007-12-12 10:16:00 +00:00
|
|
|
using namespace std;
|
2007-12-12 18:57:56 +00:00
|
|
|
using namespace lyx::support;
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2006-10-21 00:16:43 +00:00
|
|
|
namespace lyx {
|
2008-03-08 07:59:47 +00:00
|
|
|
|
|
|
|
FileName libFileSearch(QString const & dir, QString const & name,
|
2014-10-18 15:19:47 +02:00
|
|
|
QString const & ext, search_mode mode)
|
2008-03-08 07:59:47 +00:00
|
|
|
{
|
2014-10-18 15:19:47 +02:00
|
|
|
return support::libFileSearch(fromqstr(dir), fromqstr(name), fromqstr(ext), mode);
|
2008-03-08 07:59:47 +00:00
|
|
|
}
|
|
|
|
|
2008-04-20 20:32:00 +00:00
|
|
|
|
2011-03-25 01:59:34 +00:00
|
|
|
FileName imageLibFileSearch(QString & dir, QString const & name,
|
2014-10-18 15:19:47 +02:00
|
|
|
QString const & ext, search_mode mode)
|
2011-03-25 01:59:34 +00:00
|
|
|
{
|
|
|
|
string tmp = fromqstr(dir);
|
2014-10-18 15:19:47 +02:00
|
|
|
FileName fn = support::imageLibFileSearch(tmp, fromqstr(name), fromqstr(ext), mode);
|
2011-03-25 01:59:34 +00:00
|
|
|
dir = toqstr(tmp);
|
|
|
|
return fn;
|
|
|
|
}
|
|
|
|
|
2013-07-27 11:18:16 +02:00
|
|
|
namespace {
|
|
|
|
|
2014-04-04 15:54:32 -04:00
|
|
|
double locstringToDouble(QString const & str)
|
2013-07-27 11:18:16 +02:00
|
|
|
{
|
|
|
|
QLocale loc;
|
|
|
|
bool ok;
|
|
|
|
double res = loc.toDouble(str, &ok);
|
|
|
|
if (!ok) {
|
|
|
|
// Fall back to C
|
|
|
|
QLocale c(QLocale::C);
|
|
|
|
res = c.toDouble(str);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2017-07-23 13:11:54 +02:00
|
|
|
} // namespace
|
2013-07-27 11:18:16 +02:00
|
|
|
|
2011-03-25 01:59:34 +00:00
|
|
|
|
2007-11-05 22:20:23 +00:00
|
|
|
namespace frontend {
|
2006-03-05 17:24:44 +00:00
|
|
|
|
|
|
|
string widgetsToLength(QLineEdit const * input, LengthCombo const * combo)
|
|
|
|
{
|
2006-09-09 22:27:22 +00:00
|
|
|
QString const length = input->text();
|
2006-03-05 17:24:44 +00:00
|
|
|
if (length.isEmpty())
|
|
|
|
return string();
|
|
|
|
|
2006-09-09 22:27:22 +00:00
|
|
|
// Don't return unit-from-choice if the input(field) contains a unit
|
2006-03-05 17:24:44 +00:00
|
|
|
if (isValidGlueLength(fromqstr(length)))
|
|
|
|
return fromqstr(length);
|
|
|
|
|
2007-04-28 12:58:49 +00:00
|
|
|
Length::UNIT const unit = combo->currentLengthItem();
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2013-07-27 11:18:16 +02:00
|
|
|
return Length(locstringToDouble(length.trimmed()), unit).asString();
|
2006-03-05 17:24:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-28 12:58:49 +00:00
|
|
|
Length widgetsToLength(QLineEdit const * input, QComboBox const * combo)
|
2006-03-05 17:24:44 +00:00
|
|
|
{
|
2006-09-09 22:27:22 +00:00
|
|
|
QString const length = input->text();
|
2006-03-05 17:24:44 +00:00
|
|
|
if (length.isEmpty())
|
2007-04-28 12:58:49 +00:00
|
|
|
return Length();
|
2006-03-05 17:24:44 +00:00
|
|
|
|
|
|
|
// don't return unit-from-choice if the input(field) contains a unit
|
|
|
|
if (isValidGlueLength(fromqstr(length)))
|
2007-04-28 12:58:49 +00:00
|
|
|
return Length(fromqstr(length));
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2009-03-30 10:36:05 +00:00
|
|
|
Length::UNIT unit = Length::UNIT_NONE;
|
2009-01-12 06:48:12 +00:00
|
|
|
QString const item = combo->currentText();
|
|
|
|
for (int i = 0; i < num_units; i++) {
|
|
|
|
if (qt_(lyx::unit_name_gui[i]) == item) {
|
|
|
|
unit = unitFromString(unit_name[i]);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-03-05 17:24:44 +00:00
|
|
|
|
2013-07-27 11:18:16 +02:00
|
|
|
return Length(locstringToDouble(length.trimmed()), unit);
|
2006-03-05 17:24:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-04-05 14:58:15 +00:00
|
|
|
void lengthToWidgets(QLineEdit * input, LengthCombo * combo,
|
2013-04-17 11:30:25 +02:00
|
|
|
Length const & len, Length::UNIT /*defaultUnit*/)
|
2007-04-05 14:58:15 +00:00
|
|
|
{
|
2013-04-17 11:30:25 +02:00
|
|
|
if (len.empty()) {
|
|
|
|
// no length (UNIT_NONE)
|
|
|
|
combo->setCurrentItem(Length::defaultUnit());
|
|
|
|
input->setText("");
|
|
|
|
} else {
|
|
|
|
combo->setCurrentItem(len.unit());
|
|
|
|
QLocale loc;
|
|
|
|
loc.setNumberOptions(QLocale::OmitGroupSeparator);
|
2015-05-20 09:35:57 +02:00
|
|
|
input->setText(formatLocFPNumber(Length(len).value()));
|
2013-04-17 11:30:25 +02:00
|
|
|
}
|
2007-04-05 14:58:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-05 17:24:44 +00:00
|
|
|
void lengthToWidgets(QLineEdit * input, LengthCombo * combo,
|
2007-04-28 12:58:49 +00:00
|
|
|
string const & len, Length::UNIT defaultUnit)
|
2006-03-05 17:24:44 +00:00
|
|
|
{
|
|
|
|
if (len.empty()) {
|
|
|
|
// no length (UNIT_NONE)
|
|
|
|
combo->setCurrentItem(defaultUnit);
|
|
|
|
input->setText("");
|
|
|
|
} else if (!isValidLength(len) && !isStrDbl(len)) {
|
|
|
|
// use input field only for gluelengths
|
|
|
|
combo->setCurrentItem(defaultUnit);
|
|
|
|
input->setText(toqstr(len));
|
|
|
|
} else {
|
2007-04-28 12:58:49 +00:00
|
|
|
lengthToWidgets(input, combo, Length(len), defaultUnit);
|
2006-03-05 17:24:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-01-14 17:53:31 +00:00
|
|
|
void lengthToWidgets(QLineEdit * input, LengthCombo * combo,
|
|
|
|
docstring const & len, Length::UNIT defaultUnit)
|
2007-04-05 14:58:15 +00:00
|
|
|
{
|
2009-01-14 17:53:31 +00:00
|
|
|
lengthToWidgets(input, combo, to_utf8(len), defaultUnit);
|
2007-04-05 14:58:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-12 15:44:26 +00:00
|
|
|
double widgetToDouble(QLineEdit const * input)
|
|
|
|
{
|
|
|
|
QString const text = input->text();
|
|
|
|
if (text.isEmpty())
|
|
|
|
return 0.0;
|
2013-04-17 11:30:25 +02:00
|
|
|
|
2013-07-27 11:18:16 +02:00
|
|
|
return locstringToDouble(text.trimmed());
|
2009-07-12 15:44:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string widgetToDoubleStr(QLineEdit const * input)
|
|
|
|
{
|
2013-07-27 11:18:16 +02:00
|
|
|
return convert<string>(widgetToDouble(input));
|
2009-07-12 15:44:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void doubleToWidget(QLineEdit * input, double const & value, char f, int prec)
|
|
|
|
{
|
2013-07-27 11:18:16 +02:00
|
|
|
QLocale loc;
|
2009-08-08 11:48:04 +00:00
|
|
|
loc.setNumberOptions(QLocale::OmitGroupSeparator);
|
2009-07-12 15:44:26 +00:00
|
|
|
input->setText(loc.toString(value, f, prec));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void doubleToWidget(QLineEdit * input, string const & value, char f, int prec)
|
|
|
|
{
|
|
|
|
doubleToWidget(input, convert<double>(value), f, prec);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-20 09:35:57 +02:00
|
|
|
QString formatLocFPNumber(double d)
|
|
|
|
{
|
|
|
|
QString result = toqstr(formatFPNumber(d));
|
|
|
|
QLocale loc;
|
|
|
|
result.replace('.', loc.decimalPoint());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-09 16:09:50 +02:00
|
|
|
bool SortLocaleAware(QString const & lhs, QString const & rhs)
|
|
|
|
{
|
|
|
|
return QString::localeAwareCompare(lhs, rhs) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-11-16 08:19:16 +01:00
|
|
|
bool ColorSorter(ColorCode lhs, ColorCode rhs)
|
|
|
|
{
|
|
|
|
return compare_no_case(lcolor.getGUIName(lhs), lcolor.getGUIName(rhs)) < 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-15 13:06:48 +00:00
|
|
|
void setValid(QWidget * widget, bool valid)
|
|
|
|
{
|
|
|
|
if (valid) {
|
|
|
|
widget->setPalette(QPalette());
|
|
|
|
} else {
|
|
|
|
QPalette pal = widget->palette();
|
|
|
|
pal.setColor(QPalette::Active, QPalette::Foreground, QColor(255, 0, 0));
|
|
|
|
widget->setPalette(pal);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-26 22:15:50 +01:00
|
|
|
|
|
|
|
void focusAndHighlight(QAbstractItemView * w)
|
|
|
|
{
|
|
|
|
w->setFocus();
|
|
|
|
w->setCurrentIndex(w->currentIndex());
|
|
|
|
w->scrollTo(w->currentIndex());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-15 02:01:58 +02:00
|
|
|
void setMessageColour(list<QWidget *> highlighted, list<QWidget *> plain)
|
|
|
|
{
|
|
|
|
QPalette pal = QApplication::palette();
|
|
|
|
QPalette newpal(pal.color(QPalette::Active, QPalette::HighlightedText),
|
|
|
|
pal.color(QPalette::Active, QPalette::Highlight));
|
|
|
|
for (QWidget * w : highlighted)
|
|
|
|
w->setPalette(newpal);
|
|
|
|
for (QWidget * w : plain)
|
|
|
|
w->setPalette(pal);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-03 12:23:31 +01:00
|
|
|
/// wrapper to hide the change of method name to setSectionResizeMode
|
|
|
|
void setSectionResizeMode(QHeaderView * view,
|
|
|
|
int logicalIndex, QHeaderView::ResizeMode mode) {
|
|
|
|
#if (QT_VERSION >= 0x050000)
|
|
|
|
view->setSectionResizeMode(logicalIndex, mode);
|
|
|
|
#else
|
|
|
|
view->setResizeMode(logicalIndex, mode);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void setSectionResizeMode(QHeaderView * view, QHeaderView::ResizeMode mode) {
|
|
|
|
#if (QT_VERSION >= 0x050000)
|
|
|
|
view->setSectionResizeMode(mode);
|
|
|
|
#else
|
|
|
|
view->setResizeMode(mode);
|
|
|
|
#endif
|
|
|
|
}
|
2007-11-05 22:20:23 +00:00
|
|
|
} // namespace frontend
|
2007-09-15 13:06:48 +00:00
|
|
|
|
2006-11-07 18:09:55 +00:00
|
|
|
QString const qt_(char const * str, const char *)
|
2006-03-05 17:24:44 +00:00
|
|
|
{
|
2006-09-09 22:27:22 +00:00
|
|
|
return toqstr(_(str));
|
2006-03-05 17:24:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QString const qt_(string const & str)
|
|
|
|
{
|
2006-09-09 22:27:22 +00:00
|
|
|
return toqstr(_(str));
|
2006-03-05 17:24:44 +00:00
|
|
|
}
|
|
|
|
|
2008-03-08 07:59:47 +00:00
|
|
|
|
2013-05-12 18:30:40 +02:00
|
|
|
QString const qt_(QString const & qstr)
|
|
|
|
{
|
|
|
|
return toqstr(_(fromqstr(qstr)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-01 12:54:17 -04:00
|
|
|
void rescanTexStyles(string const & arg)
|
2007-11-05 22:20:23 +00:00
|
|
|
{
|
|
|
|
// Run rescan in user lyx directory
|
2007-12-12 19:57:42 +00:00
|
|
|
PathChanger p(package().user_support());
|
2012-05-01 12:54:17 -04:00
|
|
|
FileName const prog = support::libFileSearch("scripts", "TeXFiles.py");
|
2007-11-05 22:20:23 +00:00
|
|
|
Systemcall one;
|
2012-05-01 12:54:17 -04:00
|
|
|
string const command = os::python() + ' ' +
|
|
|
|
quoteName(prog.toFilesystemEncoding()) + ' ' +
|
|
|
|
arg;
|
|
|
|
int const status = one.startscript(Systemcall::Wait, command);
|
2007-11-05 22:20:23 +00:00
|
|
|
if (status == 0)
|
|
|
|
return;
|
|
|
|
// FIXME UNICODE
|
|
|
|
frontend::Alert::error(_("Could not update TeX information"),
|
2012-05-01 12:54:17 -04:00
|
|
|
bformat(_("The script `%1$s' failed."), from_utf8(prog.absFileName())));
|
2007-11-05 22:20:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-08 07:59:47 +00:00
|
|
|
QStringList texFileList(QString const & filename)
|
2007-11-05 22:20:23 +00:00
|
|
|
{
|
2008-03-08 07:59:47 +00:00
|
|
|
QStringList list;
|
|
|
|
FileName const file = libFileSearch(QString(), filename);
|
2007-11-05 22:20:23 +00:00
|
|
|
if (file.empty())
|
2008-03-08 07:59:47 +00:00
|
|
|
return list;
|
2007-11-05 22:20:23 +00:00
|
|
|
|
2007-12-02 11:55:25 +00:00
|
|
|
// FIXME Unicode.
|
2017-07-03 13:53:14 -04:00
|
|
|
vector<docstring> doclist =
|
2007-12-02 11:55:25 +00:00
|
|
|
getVectorFromString(file.fileContents("UTF-8"), from_ascii("\n"));
|
2007-11-05 22:20:23 +00:00
|
|
|
|
|
|
|
// Normalise paths like /foo//bar ==> /foo/bar
|
2008-03-08 07:59:47 +00:00
|
|
|
QSet<QString> set;
|
|
|
|
for (size_t i = 0; i != doclist.size(); ++i) {
|
2018-02-22 00:34:52 -05:00
|
|
|
QString qfile = toqstr(doclist[i]);
|
|
|
|
qfile.replace("\r", "");
|
|
|
|
while (qfile.contains("//"))
|
|
|
|
qfile.replace("//", "/");
|
|
|
|
if (!qfile.isEmpty())
|
|
|
|
set.insert(qfile);
|
2008-03-08 07:59:47 +00:00
|
|
|
}
|
|
|
|
|
2020-03-06 21:04:15 -05:00
|
|
|
// remove duplicates
|
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0))
|
|
|
|
return QList<QString>(set.begin(), set.end());
|
|
|
|
#else
|
|
|
|
return QList<QString>::fromSet(set);
|
|
|
|
#endif
|
2007-11-05 22:20:23 +00:00
|
|
|
}
|
|
|
|
|
2008-10-09 07:21:48 +00:00
|
|
|
QString const externalLineEnding(docstring const & str)
|
|
|
|
{
|
2014-08-25 20:08:59 +02:00
|
|
|
#ifdef Q_OS_MAC
|
2008-10-09 07:21:48 +00:00
|
|
|
// The MAC clipboard uses \r for lineendings, and we use \n
|
|
|
|
return toqstr(subst(str, '\n', '\r'));
|
2014-08-25 19:27:45 +02:00
|
|
|
#elif defined(Q_OS_WIN) || defined(Q_CYGWIN_WIN)
|
2008-10-09 07:21:48 +00:00
|
|
|
// Windows clipboard uses \r\n for lineendings, and we use \n
|
|
|
|
return toqstr(subst(str, from_ascii("\n"), from_ascii("\r\n")));
|
|
|
|
#else
|
|
|
|
return toqstr(str);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
docstring const internalLineEnding(QString const & str)
|
|
|
|
{
|
2017-07-03 13:53:14 -04:00
|
|
|
docstring const s = subst(qstring_to_ucs4(str),
|
2008-10-09 07:21:48 +00:00
|
|
|
from_ascii("\r\n"), from_ascii("\n"));
|
|
|
|
return subst(s, '\r', '\n');
|
|
|
|
}
|
|
|
|
|
2008-03-05 23:10:53 +00:00
|
|
|
|
|
|
|
QString internalPath(const QString & str)
|
|
|
|
{
|
|
|
|
return toqstr(os::internal_path(fromqstr(str)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-21 01:19:28 +00:00
|
|
|
QString onlyFileName(const QString & str)
|
2008-03-05 23:10:53 +00:00
|
|
|
{
|
2010-04-21 01:19:28 +00:00
|
|
|
return toqstr(support::onlyFileName(fromqstr(str)));
|
2008-03-05 23:10:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QString onlyPath(const QString & str)
|
|
|
|
{
|
|
|
|
return toqstr(support::onlyPath(fromqstr(str)));
|
|
|
|
}
|
|
|
|
|
2008-03-08 07:59:47 +00:00
|
|
|
|
|
|
|
QString changeExtension(QString const & oldname, QString const & ext)
|
|
|
|
{
|
|
|
|
return toqstr(support::changeExtension(fromqstr(oldname), fromqstr(ext)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Remove the extension from \p name
|
|
|
|
QString removeExtension(QString const & name)
|
|
|
|
{
|
|
|
|
return toqstr(support::removeExtension(fromqstr(name)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Add the extension \p ext to \p name.
|
|
|
|
Use this instead of changeExtension if you know that \p name is without
|
|
|
|
extension, because changeExtension would wrongly interpret \p name if it
|
|
|
|
contains a dot.
|
|
|
|
*/
|
|
|
|
QString addExtension(QString const & name, QString const & ext)
|
|
|
|
{
|
|
|
|
return toqstr(support::addExtension(fromqstr(name), fromqstr(ext)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the extension of the file (not including the .)
|
|
|
|
QString getExtension(QString const & name)
|
|
|
|
{
|
|
|
|
return toqstr(support::getExtension(fromqstr(name)));
|
|
|
|
}
|
|
|
|
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
/** Convert relative path into absolute path based on a basepath.
|
|
|
|
If relpath is absolute, just use that.
|
|
|
|
If basepath doesn't exist use CWD.
|
|
|
|
*/
|
|
|
|
QString makeAbsPath(QString const & relpath, QString const & base)
|
|
|
|
{
|
|
|
|
return toqstr(support::makeAbsPath(fromqstr(relpath),
|
2010-04-21 01:19:09 +00:00
|
|
|
fromqstr(base)).absFileName());
|
2008-04-20 19:56:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
|
|
// FileFilterList
|
|
|
|
//
|
|
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/** Given a string such as
|
|
|
|
* "<glob> <glob> ... *.{abc,def} <glob>",
|
Run codespell on src/frontends
Command was:
codespell -w -i 3 -S Makefile.in -L mathed,afe,tthe,ue,fro,uint,larg,alph,te,thes,alle,Claus,pres,pass-thru src/frontends/
2020-06-26 00:04:31 +02:00
|
|
|
* convert the csh-style brace expressions:
|
2008-04-20 19:56:01 +00:00
|
|
|
* "<glob> <glob> ... *.abc *.def <glob>".
|
|
|
|
* Requires no system support, so should work equally on Unix, Mac, Win32.
|
|
|
|
*/
|
|
|
|
static string const convert_brace_glob(string const & glob)
|
|
|
|
{
|
|
|
|
// Matches " *.{abc,def,ghi}", storing "*." as group 1 and
|
2011-09-13 20:42:23 +00:00
|
|
|
// "abc,def,ghi" as group 2, while allowing spaces in group 2.
|
|
|
|
static lyx::regex const glob_re(" *([^ {]*)\\{([^}]+)\\}");
|
|
|
|
// Matches "abc" and "abc,", storing "abc" as group 1,
|
|
|
|
// while ignoring surrounding spaces.
|
|
|
|
static lyx::regex const block_re(" *([^ ,}]+) *,? *");
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
string pattern;
|
|
|
|
|
|
|
|
string::const_iterator it = glob.begin();
|
|
|
|
string::const_iterator const end = glob.end();
|
|
|
|
while (true) {
|
2010-06-29 17:09:40 +00:00
|
|
|
match_results<string::const_iterator> what;
|
|
|
|
if (!regex_search(it, end, what, glob_re)) {
|
2008-04-20 19:56:01 +00:00
|
|
|
// Ensure that no information is lost.
|
|
|
|
pattern += string(it, end);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Everything from the start of the input to
|
|
|
|
// the start of the match.
|
|
|
|
pattern += string(what[-1].first, what[-1].second);
|
|
|
|
|
|
|
|
// Given " *.{abc,def}", head == "*." and tail == "abc,def".
|
|
|
|
string const head = string(what[1].first, what[1].second);
|
|
|
|
string const tail = string(what[2].first, what[2].second);
|
|
|
|
|
|
|
|
// Split the ','-separated chunks of tail so that
|
|
|
|
// $head{$chunk1,$chunk2} becomes "$head$chunk1 $head$chunk2".
|
|
|
|
string const fmt = " " + head + "$1";
|
2010-06-29 17:09:40 +00:00
|
|
|
pattern += regex_replace(tail, block_re, fmt);
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
// Increment the iterator to the end of the match.
|
|
|
|
it += distance(it, what[0].second);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pattern;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct Filter
|
|
|
|
{
|
|
|
|
/* \param description text describing the filters.
|
|
|
|
* \param one or more wildcard patterns, separated by
|
|
|
|
* whitespace.
|
|
|
|
*/
|
|
|
|
Filter(docstring const & description, std::string const & globs);
|
|
|
|
|
|
|
|
docstring const & description() const { return desc_; }
|
|
|
|
|
|
|
|
QString toString() const;
|
|
|
|
|
|
|
|
docstring desc_;
|
|
|
|
std::vector<std::string> globs_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Filter::Filter(docstring const & description, string const & globs)
|
|
|
|
: desc_(description)
|
|
|
|
{
|
|
|
|
// Given "<glob> <glob> ... *.{abc,def} <glob>", expand to
|
|
|
|
// "<glob> <glob> ... *.abc *.def <glob>"
|
|
|
|
string const expanded_globs = convert_brace_glob(globs);
|
|
|
|
|
|
|
|
// Split into individual globs.
|
2016-05-19 15:28:39 +02:00
|
|
|
globs_ = getVectorFromString(expanded_globs, " ");
|
2008-04-20 19:56:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
QString Filter::toString() const
|
|
|
|
{
|
|
|
|
QString s;
|
|
|
|
|
2011-09-13 20:42:55 +00:00
|
|
|
bool const has_description = !desc_.empty();
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
if (has_description) {
|
|
|
|
s += toqstr(desc_);
|
|
|
|
s += " (";
|
|
|
|
}
|
|
|
|
|
2016-05-19 15:28:39 +02:00
|
|
|
s += toqstr(getStringFromVector(globs_, " "));
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
if (has_description)
|
|
|
|
s += ')';
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** \c FileFilterList parses a Qt-style list of available file filters
|
|
|
|
* to generate the corresponding vector.
|
|
|
|
* For example "TeX documents (*.tex);;LyX Documents (*.lyx)"
|
|
|
|
* will be parsed to fill a vector of size 2, whilst "*.{p[bgp]m} *.pdf"
|
|
|
|
* will result in a vector of size 1 in which the description field is empty.
|
|
|
|
*/
|
|
|
|
struct FileFilterList
|
|
|
|
{
|
|
|
|
// FIXME UNICODE: globs_ should be unicode...
|
|
|
|
/** \param qt_style_filter a list of available file filters.
|
|
|
|
* Eg. "TeX documents (*.tex);;LyX Documents (*.lyx)".
|
|
|
|
* The "All files (*)" filter is always added to the list.
|
|
|
|
*/
|
|
|
|
explicit FileFilterList(docstring const & qt_style_filter =
|
|
|
|
docstring());
|
|
|
|
|
|
|
|
typedef std::vector<Filter>::size_type size_type;
|
|
|
|
|
|
|
|
bool empty() const { return filters_.empty(); }
|
|
|
|
size_type size() const { return filters_.size(); }
|
|
|
|
Filter & operator[](size_type i) { return filters_[i]; }
|
|
|
|
Filter const & operator[](size_type i) const { return filters_[i]; }
|
|
|
|
|
|
|
|
void parse_filter(std::string const & filter);
|
|
|
|
std::vector<Filter> filters_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
FileFilterList::FileFilterList(docstring const & qt_style_filter)
|
|
|
|
{
|
|
|
|
// FIXME UNICODE
|
|
|
|
string const filter = to_utf8(qt_style_filter)
|
|
|
|
+ (qt_style_filter.empty() ? string() : ";;")
|
|
|
|
+ to_utf8(_("All Files "))
|
2017-07-03 13:53:14 -04:00
|
|
|
#if defined(_WIN32)
|
2008-04-20 19:56:01 +00:00
|
|
|
+ ("(*.*)");
|
|
|
|
#else
|
|
|
|
+ ("(*)");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Split data such as "TeX documents (*.tex);;LyX Documents (*.lyx)"
|
|
|
|
// into individual filters.
|
2010-06-29 17:09:40 +00:00
|
|
|
static lyx::regex const separator_re(";;");
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
string::const_iterator it = filter.begin();
|
|
|
|
string::const_iterator const end = filter.end();
|
|
|
|
while (true) {
|
2010-06-29 17:09:40 +00:00
|
|
|
match_results<string::const_iterator> what;
|
2008-04-20 19:56:01 +00:00
|
|
|
|
2010-06-29 17:09:40 +00:00
|
|
|
if (!lyx::regex_search(it, end, what, separator_re)) {
|
2008-04-20 19:56:01 +00:00
|
|
|
parse_filter(string(it, end));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Everything from the start of the input to
|
|
|
|
// the start of the match.
|
2016-09-05 22:49:59 +02:00
|
|
|
parse_filter(string(it, what[0].first));
|
2008-04-20 19:56:01 +00:00
|
|
|
|
|
|
|
// Increment the iterator to the end of the match.
|
|
|
|
it += distance(it, what[0].second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FileFilterList::parse_filter(string const & filter)
|
|
|
|
{
|
2011-09-13 20:42:55 +00:00
|
|
|
// Matches "TeX documents (plain) (*.tex)",
|
|
|
|
// storing "TeX documents (plain) " as group 1 and "*.tex" as group 2.
|
|
|
|
static lyx::regex const filter_re("(.*)\\(([^()]+)\\) *$");
|
2008-04-20 19:56:01 +00:00
|
|
|
|
2010-06-29 17:09:40 +00:00
|
|
|
match_results<string::const_iterator> what;
|
|
|
|
if (!lyx::regex_search(filter, what, filter_re)) {
|
2008-04-20 19:56:01 +00:00
|
|
|
// Just a glob, no description.
|
|
|
|
filters_.push_back(Filter(docstring(), trim(filter)));
|
|
|
|
} else {
|
|
|
|
// FIXME UNICODE
|
|
|
|
docstring const desc = from_utf8(string(what[1].first, what[1].second));
|
|
|
|
string const globs = string(what[2].first, what[2].second);
|
|
|
|
filters_.push_back(Filter(trim(desc), trim(globs)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** \returns the equivalent of the string passed in
|
|
|
|
* although any brace expressions are expanded.
|
|
|
|
* (E.g. "*.{png,jpg}" -> "*.png *.jpg")
|
|
|
|
*/
|
|
|
|
QStringList fileFilters(QString const & desc)
|
|
|
|
{
|
|
|
|
// we have: "*.{gif,png,jpg,bmp,pbm,ppm,tga,tif,xpm,xbm}"
|
|
|
|
// but need: "*.cpp;*.cc;*.C;*.cxx;*.c++"
|
|
|
|
FileFilterList filters(qstring_to_ucs4(desc));
|
2008-05-23 13:34:16 +00:00
|
|
|
//LYXERR0("DESC: " << desc);
|
2008-04-20 19:56:01 +00:00
|
|
|
QStringList list;
|
|
|
|
for (size_t i = 0; i != filters.filters_.size(); ++i) {
|
|
|
|
QString f = filters.filters_[i].toString();
|
2008-05-23 13:34:16 +00:00
|
|
|
//LYXERR0("FILTER: " << f);
|
2008-04-20 19:56:01 +00:00
|
|
|
list.append(f);
|
|
|
|
}
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2008-05-16 13:49:49 +00:00
|
|
|
|
2016-06-05 21:35:35 +02:00
|
|
|
QString formatToolTip(QString text, int em)
|
|
|
|
{
|
|
|
|
// 1. QTooltip activates word wrapping only if mightBeRichText()
|
|
|
|
// is true. So we convert the text to rich text.
|
|
|
|
//
|
|
|
|
// 2. The default width is way too small. Setting the width is tricky; first
|
|
|
|
// one has to compute the ideal width, and then force it with special
|
|
|
|
// html markup.
|
|
|
|
|
|
|
|
// do nothing if empty or already formatted
|
|
|
|
if (text.isEmpty() || text.startsWith(QString("<html>")))
|
|
|
|
return text;
|
|
|
|
// Convert to rich text if it is not already
|
|
|
|
if (!Qt::mightBeRichText(text))
|
|
|
|
text = Qt::convertFromPlainText(text, Qt::WhiteSpaceNormal);
|
|
|
|
// Compute desired width in pixels
|
|
|
|
QFont const font = QToolTip::font();
|
|
|
|
int const px_width = em * QFontMetrics(font).width("M");
|
|
|
|
// Determine the ideal width of the tooltip
|
2017-02-25 00:32:58 +01:00
|
|
|
QTextDocument td("");
|
|
|
|
td.setHtml(text);
|
|
|
|
td.setDefaultFont(QToolTip::font());
|
2018-11-04 13:53:36 -10:00
|
|
|
td.setDocumentMargin(0);
|
2017-02-25 00:32:58 +01:00
|
|
|
td.setTextWidth(px_width);
|
|
|
|
double best_width = td.idealWidth();
|
2016-06-05 21:35:35 +02:00
|
|
|
// Set the line wrapping with appropriate width
|
|
|
|
return QString("<html><body><table><tr>"
|
|
|
|
"<td align=justify width=%1>%2</td>"
|
|
|
|
"</tr></table></body></html>")
|
|
|
|
.arg(QString::number(int(best_width) + 1), text);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-04 18:28:03 +01:00
|
|
|
QString qtHtmlToPlainText(QString const & html)
|
|
|
|
{
|
|
|
|
if (!Qt::mightBeRichText(html))
|
|
|
|
return html;
|
|
|
|
QTextDocument td;
|
|
|
|
td.setHtml(html);
|
|
|
|
return td.toPlainText();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-10-21 00:16:43 +00:00
|
|
|
} // namespace lyx
|