413 lines
11 KiB
C++
413 lines
11 KiB
C++
// For license of this file, see <project-root-folder>/LICENSE.md.
|
|
|
|
#include "miscellaneous/textfactory.h"
|
|
|
|
#include "3rd-party/boolinq/boolinq.h"
|
|
#include "3rd-party/sc/simplecrypt.h"
|
|
#include "definitions/definitions.h"
|
|
#include "exceptions/applicationexception.h"
|
|
#include "miscellaneous/application.h"
|
|
#include "miscellaneous/iofactory.h"
|
|
#include "miscellaneous/settings.h"
|
|
|
|
#include <QDir>
|
|
#include <QLocale>
|
|
#include <QRandomGenerator64>
|
|
#include <QString>
|
|
#include <QStringList>
|
|
#include <QTextDocument>
|
|
|
|
quint64 TextFactory::s_encryptionKey = 0x0;
|
|
|
|
TextFactory::TextFactory() = default;
|
|
|
|
QString TextFactory::extractUsernameFromEmail(const QString& email_address) {
|
|
const int zav = email_address.indexOf('@');
|
|
|
|
if (zav >= 0) {
|
|
return email_address.mid(0, zav);
|
|
}
|
|
else {
|
|
return email_address;
|
|
}
|
|
}
|
|
|
|
QColor TextFactory::generateColorFromText(const QString& text) {
|
|
quint32 color = 0;
|
|
|
|
for (const QChar chr : text) {
|
|
color += chr.unicode();
|
|
}
|
|
|
|
// NOTE: https://martin.ankerl.com/2009/12/09/how-to-create-random-colors-programmatically
|
|
int hue = color % 360;
|
|
|
|
return QColor::fromHsv(hue, 200, 240);
|
|
}
|
|
|
|
QColor TextFactory::generateRandomColor() {
|
|
int hue = QRandomGenerator::global()->generate() % 360;
|
|
auto clr = QColor::fromHsv(hue, 200, 240);
|
|
|
|
return clr;
|
|
}
|
|
|
|
int TextFactory::stringHeight(const QString& string, const QFontMetrics& metrics) {
|
|
const int count_lines = string.split(QL1C('\n')).size();
|
|
|
|
return metrics.height() * count_lines;
|
|
}
|
|
|
|
int TextFactory::stringWidth(const QString& string, const QFontMetrics& metrics) {
|
|
const QStringList lines = string.split(QL1C('\n'));
|
|
int width = 0;
|
|
|
|
for (const QString& line : lines) {
|
|
int line_width = metrics.horizontalAdvance(line);
|
|
|
|
if (line_width > width) {
|
|
width = line_width;
|
|
}
|
|
}
|
|
|
|
return width;
|
|
}
|
|
|
|
bool TextFactory::couldBeHtml(const QString& string) {
|
|
const QString sstring = string.simplified();
|
|
|
|
return sstring.startsWith(QL1S("<!")) || sstring.startsWith(QL1S("<html")) || sstring.startsWith(QL1S("<figure")) ||
|
|
sstring.startsWith(QL1S("<article")) || sstring.startsWith(QL1S("<details")) || Qt::mightBeRichText(sstring);
|
|
}
|
|
|
|
QDateTime TextFactory::parseDateTime(const QString& date_time, QString* used_dt_format) {
|
|
QString input_date = date_time.simplified().replace(QSL("GMT"), QSL("UTC"));
|
|
|
|
if (input_date.isEmpty()) {
|
|
return QDateTime();
|
|
}
|
|
|
|
QLocale locale(QLocale::Language::C);
|
|
QDateTime dt;
|
|
QStringList date_patterns = dateTimePatterns(true);
|
|
|
|
if (used_dt_format != nullptr && !used_dt_format->isEmpty()) {
|
|
date_patterns.prepend(*used_dt_format);
|
|
}
|
|
|
|
// QDateTime dt1 = locale.toDateTime("GMT", "t");
|
|
// QString dt2 = dt1.toString();
|
|
|
|
for (const QString& pattern : std::as_const(date_patterns)) {
|
|
#if QT_VERSION >= 0x060700 // Qt >= 6.7.0
|
|
dt = locale.toDateTime(input_date, pattern, 2000);
|
|
#else
|
|
dt = locale.toDateTime(input_date, pattern);
|
|
#endif
|
|
|
|
if (dt.isValid()) {
|
|
// Make sure that this date/time is considered UTC.
|
|
dt = dt.toUTC();
|
|
|
|
if (used_dt_format != nullptr) {
|
|
used_dt_format->clear();
|
|
used_dt_format->append(pattern);
|
|
}
|
|
|
|
return dt;
|
|
}
|
|
}
|
|
|
|
qCriticalNN << LOGSEC_CORE << "Date/time string NOT recognized:" << QUOTE_W_SPACE_DOT(input_date);
|
|
return QDateTime();
|
|
}
|
|
|
|
QDateTime TextFactory::parseDateTime(qint64 milis_from_epoch) {
|
|
return QDateTime::fromMSecsSinceEpoch(milis_from_epoch, Qt::TimeSpec::UTC);
|
|
}
|
|
|
|
QStringList TextFactory::dateTimePatterns(bool with_tzs) {
|
|
QStringList pat;
|
|
|
|
pat << QSL("yyyy-MM-ddTHH:mm:ss");
|
|
pat << QSL("yyyy-MM-ddTHH:mm:ss.zzz");
|
|
pat << QSL("yyyy-MM-ddThh:mm:ss");
|
|
pat << QSL("yyyy-MM-dd HH:mm:ss.z");
|
|
|
|
pat << QSL("yyyy-MM-ddThh:mm");
|
|
|
|
pat << QSL("yyyyMMddThhmmss");
|
|
pat << QSL("yyyyMMdd");
|
|
pat << QSL("yyyy");
|
|
|
|
pat << QSL("yyyy-MM-dd");
|
|
pat << QSL("yyyy-MM");
|
|
|
|
pat << QSL("MMM dd yyyy hh:mm:ss");
|
|
pat << QSL("MMM d yyyy hh:mm:ss");
|
|
|
|
pat << QSL("ddd, dd MMM yyyy HH:mm:ss");
|
|
pat << QSL("ddd, dd MMM yyyy HH:mm");
|
|
pat << QSL("ddd, dd MMM yy HH:mm:ss");
|
|
pat << QSL("ddd, d MMM yyyy HH:mm:ss");
|
|
|
|
pat << QSL("ddd, MM/dd/yyyy - HH:mm");
|
|
|
|
pat << QSL("dd MMM yyyy hh:mm:ss");
|
|
pat << QSL("dd MMM yyyy");
|
|
|
|
pat << QSL("d MMM yyyy HH:mm:ss");
|
|
|
|
pat << QSL("hh:mm:ss");
|
|
pat << QSL("h:m:s");
|
|
pat << QSL("h:mm");
|
|
pat << QSL("H:mm");
|
|
pat << QSL("h:m");
|
|
pat << QSL("h.m");
|
|
|
|
if (with_tzs) {
|
|
for (int i = 0; i < pat.size(); i += 3) {
|
|
QString base_pattern = pat.value(i);
|
|
|
|
pat.insert(i + 1, base_pattern + QSL("t"));
|
|
pat.insert(i + 2, base_pattern + QSL(" t"));
|
|
}
|
|
}
|
|
|
|
return pat;
|
|
}
|
|
|
|
QString TextFactory::encrypt(const QString& text, quint64 key) {
|
|
return SimpleCrypt(key == 0 ? initializeSecretEncryptionKey() : key).encryptToString(text);
|
|
}
|
|
|
|
QString TextFactory::decrypt(const QString& text, quint64 key) {
|
|
return SimpleCrypt(key == 0 ? initializeSecretEncryptionKey() : key).decryptToString(text);
|
|
}
|
|
|
|
QString TextFactory::newline() {
|
|
#if defined(Q_OS_WIN)
|
|
return QSL("\r\n");
|
|
#elif defined(Q_OS_MACOS)
|
|
return QSL("\r");
|
|
#else
|
|
return QSL("\n");
|
|
#endif
|
|
}
|
|
|
|
QString TextFactory::capitalizeFirstLetter(const QString& sts) {
|
|
if (sts.isEmpty()) {
|
|
return sts;
|
|
}
|
|
else {
|
|
return sts[0].toUpper() + sts.mid(1);
|
|
}
|
|
}
|
|
|
|
enum class TokenState {
|
|
// We are not inside argument, we are between arguments.
|
|
Normal,
|
|
|
|
// We have detected escape "\" character coming from double-quoted argument.
|
|
EscapedFromDoubleQuotes,
|
|
|
|
// We have detected escape "\" character coming from spaced argument.
|
|
EscapedFromSpaced,
|
|
|
|
// We are inside argument which was separated by spaces.
|
|
InsideArgSpaced,
|
|
|
|
// We are inside argument.
|
|
InsideArgDoubleQuotes,
|
|
|
|
// We are inside argument, do not evaluate anything, just take it all
|
|
// as arw text.
|
|
InsideArgSingleQuotes
|
|
};
|
|
|
|
QStringList TextFactory::tokenizeProcessArguments(const QString& command) {
|
|
// Each argument containing spaces must be enclosed with single '' or double "" quotes.
|
|
// Some characters must be escaped with \ to keep their textual values as
|
|
// long as double-quoted argument is used.
|
|
|
|
if (command.isEmpty()) {
|
|
return {};
|
|
}
|
|
|
|
// We append space to end of command to make sure that
|
|
// ending space-separated argument is processed.
|
|
QString my_command = command + u' ';
|
|
|
|
TokenState state = TokenState::Normal;
|
|
QStringList args;
|
|
QString arg;
|
|
|
|
for (QChar chr : my_command) {
|
|
switch (state) {
|
|
case TokenState::Normal: {
|
|
switch (chr.unicode()) {
|
|
case u'"':
|
|
// We start double-quoted argument.
|
|
state = TokenState::InsideArgDoubleQuotes;
|
|
continue;
|
|
|
|
case u'\'':
|
|
// We start single-quoted argument.
|
|
state = TokenState::InsideArgSingleQuotes;
|
|
continue;
|
|
|
|
case u' ':
|
|
// Whitespace, just go on.
|
|
continue;
|
|
|
|
default:
|
|
// We found some actual text which marks
|
|
// beginning of argument, we assume spaced argument.
|
|
arg.append(chr);
|
|
state = TokenState::InsideArgSpaced;
|
|
continue;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case TokenState::EscapedFromDoubleQuotes: {
|
|
// Previous character was "\".
|
|
arg.append(chr);
|
|
state = TokenState::InsideArgDoubleQuotes;
|
|
break;
|
|
}
|
|
|
|
case TokenState::EscapedFromSpaced: {
|
|
// Previous character was "\".
|
|
arg.append(chr);
|
|
state = TokenState::InsideArgSpaced;
|
|
break;
|
|
}
|
|
|
|
case TokenState::InsideArgSpaced: {
|
|
switch (chr.unicode()) {
|
|
// NOTE: Probably disable escaping in spaced arguments to provide simpler UX?
|
|
/*
|
|
case u'\\':
|
|
// We found escaped!
|
|
state = TokenState::EscapedFromSpaced;
|
|
continue;
|
|
*/
|
|
|
|
case u' ':
|
|
// We need to end this argument.
|
|
args.append(arg);
|
|
arg.clear();
|
|
state = TokenState::Normal;
|
|
continue;
|
|
|
|
default:
|
|
arg.append(chr);
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case TokenState::InsideArgDoubleQuotes: {
|
|
switch (chr.unicode()) {
|
|
case u'\\':
|
|
// We found escaped!
|
|
state = TokenState::EscapedFromDoubleQuotes;
|
|
continue;
|
|
|
|
case u'"':
|
|
// We need to end this argument.
|
|
args.append(arg);
|
|
arg.clear();
|
|
state = TokenState::Normal;
|
|
continue;
|
|
|
|
default:
|
|
arg.append(chr);
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case TokenState::InsideArgSingleQuotes: {
|
|
switch (chr.unicode()) {
|
|
case u'\'':
|
|
// We need to end this argument.
|
|
args.append(arg);
|
|
arg.clear();
|
|
state = TokenState::Normal;
|
|
continue;
|
|
|
|
default:
|
|
arg.append(chr);
|
|
break;
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
switch (state) {
|
|
case TokenState::EscapedFromSpaced:
|
|
case TokenState::EscapedFromDoubleQuotes:
|
|
throw ApplicationException(QObject::tr("escape sequence not completed"));
|
|
break;
|
|
|
|
case TokenState::InsideArgDoubleQuotes:
|
|
throw ApplicationException(QObject::tr("closing \" is missing"));
|
|
break;
|
|
|
|
case TokenState::InsideArgSingleQuotes:
|
|
throw ApplicationException(QObject::tr("closing ' is missing"));
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return args;
|
|
}
|
|
|
|
QString TextFactory::shorten(const QString& input, int text_length_limit) {
|
|
if (input.size() > text_length_limit) {
|
|
return input.left(text_length_limit - ELLIPSIS_LENGTH) + QString(ELLIPSIS_LENGTH, QL1C('.'));
|
|
}
|
|
else {
|
|
return input;
|
|
}
|
|
}
|
|
|
|
quint64 TextFactory::initializeSecretEncryptionKey() {
|
|
if (s_encryptionKey == 0x0) {
|
|
// Check if file with encryption key exists.
|
|
QString encryption_file_path = qApp->settings()->pathName() + QDir::separator() + ENCRYPTION_FILE_NAME;
|
|
|
|
try {
|
|
s_encryptionKey = quint64(QString(IOFactory::readFile(encryption_file_path)).toULongLong());
|
|
}
|
|
catch (ApplicationException&) {
|
|
// Well, key does not exist or is invalid, generate and save one.
|
|
s_encryptionKey = generateSecretEncryptionKey();
|
|
|
|
try {
|
|
IOFactory::writeFile(encryption_file_path, QString::number(s_encryptionKey).toLocal8Bit());
|
|
}
|
|
catch (ApplicationException& ex) {
|
|
qCriticalNN << LOGSEC_CORE << "Failed to write newly generated encryption key to file, error"
|
|
<< QUOTE_W_SPACE_DOT(ex.message())
|
|
<< " Now, your passwords won't be readable after you start this application again.";
|
|
}
|
|
}
|
|
}
|
|
|
|
return s_encryptionKey;
|
|
}
|
|
|
|
quint64 TextFactory::generateSecretEncryptionKey() {
|
|
return QRandomGenerator64::global()->generate();
|
|
}
|