2015-03-25 10:56:14 +00:00
|
|
|
#include "search.hpp"
|
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <QModelIndex>
|
2023-01-10 18:12:13 +00:00
|
|
|
#include <QRegularExpression>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <QString>
|
|
|
|
|
|
|
|
#include <memory>
|
2015-03-25 10:56:14 +00:00
|
|
|
#include <sstream>
|
|
|
|
#include <stdexcept>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <apps/opencs/model/world/columns.hpp>
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
#include "../doc/document.hpp"
|
|
|
|
#include "../doc/messages.hpp"
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
#include "../world/columnbase.hpp"
|
|
|
|
#include "../world/commands.hpp"
|
|
|
|
#include "../world/idtablebase.hpp"
|
|
|
|
#include "../world/universalid.hpp"
|
2015-03-25 10:56:14 +00:00
|
|
|
|
|
|
|
void CSMTools::Search::searchTextCell(const CSMWorld::IdTableBase* model, const QModelIndex& index,
|
2015-04-16 16:50:22 +00:00
|
|
|
const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
|
|
|
// using QString here for easier handling of case folding.
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
QString search = QString::fromUtf8(mText.c_str());
|
|
|
|
QString text = model->data(index).toString();
|
|
|
|
|
|
|
|
int pos = 0;
|
|
|
|
|
2018-07-04 19:03:54 +00:00
|
|
|
Qt::CaseSensitivity caseSensitivity = mCase ? Qt::CaseSensitive : Qt::CaseInsensitive;
|
|
|
|
while ((pos = text.indexOf(search, pos, caseSensitivity)) != -1)
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
|
|
|
std::ostringstream hint;
|
2015-03-29 13:28:31 +00:00
|
|
|
hint << (writable ? 'R' : 'r') << ": " << model->getColumnId(index.column()) << " " << pos << " "
|
|
|
|
<< search.length();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-29 16:16:43 +00:00
|
|
|
messages.add(id, formatDescription(text, pos, search.length()).toUtf8().data(), hint.str());
|
2015-03-25 10:56:14 +00:00
|
|
|
|
|
|
|
pos += search.length();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::Search::searchRegExCell(const CSMWorld::IdTableBase* model, const QModelIndex& index,
|
2015-04-16 16:50:22 +00:00
|
|
|
const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2023-01-13 07:59:03 +00:00
|
|
|
// TODO: verify regular expression before starting a search
|
|
|
|
if (!mRegExp.isValid())
|
|
|
|
return;
|
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
QString text = model->data(index).toString();
|
|
|
|
|
2023-01-13 07:28:49 +00:00
|
|
|
QRegularExpressionMatchIterator i = mRegExp.globalMatch(text);
|
|
|
|
while (i.hasNext())
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2023-01-13 07:28:49 +00:00
|
|
|
QRegularExpressionMatch match = i.next();
|
|
|
|
|
|
|
|
int pos = match.capturedStart();
|
2023-01-10 18:12:13 +00:00
|
|
|
int length = match.capturedLength();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
std::ostringstream hint;
|
2015-04-16 16:50:22 +00:00
|
|
|
hint << (writable ? 'R' : 'r') << ": " << model->getColumnId(index.column()) << " " << pos << " " << length;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-29 16:16:43 +00:00
|
|
|
messages.add(id, formatDescription(text, pos, length).toUtf8().data(), hint.str());
|
2015-03-25 10:56:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::Search::searchRecordStateCell(const CSMWorld::IdTableBase* model, const QModelIndex& index,
|
2015-04-16 16:50:22 +00:00
|
|
|
const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2015-04-16 16:50:22 +00:00
|
|
|
if (writable)
|
|
|
|
throw std::logic_error("Record state can not be modified by search and replace");
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
int data = model->data(index).toInt();
|
|
|
|
|
|
|
|
if (data == mValue)
|
|
|
|
{
|
2019-01-22 06:08:48 +00:00
|
|
|
std::vector<std::pair<int, std::string>> states
|
2015-03-25 10:56:14 +00:00
|
|
|
= CSMWorld::Columns::getEnums(CSMWorld::Columns::ColumnId_Modification);
|
|
|
|
|
2019-01-20 11:46:19 +00:00
|
|
|
const std::string hint = "r: " + std::to_string(model->getColumnId(index.column()));
|
2019-01-22 06:08:48 +00:00
|
|
|
messages.add(id, states.at(data).second, hint);
|
2015-03-25 10:56:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-29 16:16:43 +00:00
|
|
|
QString CSMTools::Search::formatDescription(const QString& description, int pos, int length) const
|
2015-03-27 18:10:45 +00:00
|
|
|
{
|
2015-03-30 10:52:08 +00:00
|
|
|
QString text(description);
|
2015-03-29 16:16:43 +00:00
|
|
|
|
2015-03-30 10:52:08 +00:00
|
|
|
// split
|
|
|
|
QString highlight = flatten(text.mid(pos, length));
|
2015-03-30 20:30:33 +00:00
|
|
|
QString before = flatten(mPaddingBefore >= pos ? text.mid(0, pos) : text.mid(pos - mPaddingBefore, mPaddingBefore));
|
|
|
|
QString after = flatten(text.mid(pos + length, mPaddingAfter));
|
2015-03-30 10:52:08 +00:00
|
|
|
|
2015-03-31 11:02:12 +00:00
|
|
|
// compensate for Windows nonsense
|
|
|
|
text.remove('\r');
|
|
|
|
|
2015-03-30 10:52:08 +00:00
|
|
|
// join
|
|
|
|
text = before + "<b>" + highlight + "</b>" + after;
|
|
|
|
|
2015-03-29 16:16:43 +00:00
|
|
|
// improve layout for single line display
|
2015-03-30 10:52:08 +00:00
|
|
|
text.replace("\n", "<CR>");
|
2015-03-29 16:16:43 +00:00
|
|
|
text.replace('\t', ' ');
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-30 10:52:08 +00:00
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString CSMTools::Search::flatten(const QString& text) const
|
|
|
|
{
|
|
|
|
QString flat(text);
|
|
|
|
|
|
|
|
flat.replace("&", "&");
|
|
|
|
flat.replace("<", "<");
|
|
|
|
|
|
|
|
return flat;
|
2015-03-27 18:10:45 +00:00
|
|
|
}
|
|
|
|
|
2018-07-04 19:03:54 +00:00
|
|
|
CSMTools::Search::Search()
|
|
|
|
: mType(Type_None)
|
|
|
|
, mValue(0)
|
|
|
|
, mCase(false)
|
|
|
|
, mIdColumn(0)
|
|
|
|
, mTypeColumn(0)
|
2016-01-03 17:20:34 +00:00
|
|
|
, mPaddingBefore(10)
|
|
|
|
, mPaddingAfter(10)
|
|
|
|
{
|
|
|
|
}
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2018-07-04 19:03:54 +00:00
|
|
|
CSMTools::Search::Search(Type type, bool caseSensitive, const std::string& value)
|
|
|
|
: mType(type)
|
|
|
|
, mText(value)
|
|
|
|
, mValue(0)
|
|
|
|
, mCase(caseSensitive)
|
|
|
|
, mIdColumn(0)
|
|
|
|
, mTypeColumn(0)
|
|
|
|
, mPaddingBefore(10)
|
|
|
|
, mPaddingAfter(10)
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2015-03-28 10:54:32 +00:00
|
|
|
if (type != Type_Text && type != Type_Id)
|
2015-03-25 10:56:14 +00:00
|
|
|
throw std::logic_error("Invalid search parameter (string)");
|
|
|
|
}
|
|
|
|
|
2023-01-10 18:12:13 +00:00
|
|
|
CSMTools::Search::Search(Type type, bool caseSensitive, const QRegularExpression& value)
|
2018-07-04 19:03:54 +00:00
|
|
|
: mType(type)
|
|
|
|
, mRegExp(value)
|
|
|
|
, mValue(0)
|
|
|
|
, mCase(caseSensitive)
|
|
|
|
, mIdColumn(0)
|
|
|
|
, mTypeColumn(0)
|
|
|
|
, mPaddingBefore(10)
|
|
|
|
, mPaddingAfter(10)
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2023-01-10 18:12:13 +00:00
|
|
|
mRegExp.setPatternOptions(mCase ? QRegularExpression::NoPatternOption : QRegularExpression::CaseInsensitiveOption);
|
2015-03-28 10:54:32 +00:00
|
|
|
if (type != Type_TextRegEx && type != Type_IdRegEx)
|
2015-03-25 10:56:14 +00:00
|
|
|
throw std::logic_error("Invalid search parameter (RegExp)");
|
|
|
|
}
|
|
|
|
|
2018-07-04 19:03:54 +00:00
|
|
|
CSMTools::Search::Search(Type type, bool caseSensitive, int value)
|
|
|
|
: mType(type)
|
|
|
|
, mValue(value)
|
|
|
|
, mCase(caseSensitive)
|
|
|
|
, mIdColumn(0)
|
|
|
|
, mTypeColumn(0)
|
|
|
|
, mPaddingBefore(10)
|
|
|
|
, mPaddingAfter(10)
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
|
|
|
if (type != Type_RecordState)
|
|
|
|
throw std::logic_error("invalid search parameter (int)");
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::Search::configure(const CSMWorld::IdTableBase* model)
|
|
|
|
{
|
|
|
|
mColumns.clear();
|
|
|
|
|
|
|
|
int columns = model->columnCount();
|
|
|
|
|
|
|
|
for (int i = 0; i < columns; ++i)
|
|
|
|
{
|
|
|
|
CSMWorld::ColumnBase::Display display = static_cast<CSMWorld::ColumnBase::Display>(
|
|
|
|
model->headerData(i, Qt::Horizontal, static_cast<int>(CSMWorld::ColumnBase::Role_Display)).toInt());
|
|
|
|
|
|
|
|
bool consider = false;
|
|
|
|
|
|
|
|
switch (mType)
|
|
|
|
{
|
|
|
|
case Type_Text:
|
|
|
|
case Type_TextRegEx:
|
|
|
|
|
|
|
|
if (CSMWorld::ColumnBase::isText(display) || CSMWorld::ColumnBase::isScript(display))
|
|
|
|
{
|
|
|
|
consider = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-28 10:54:32 +00:00
|
|
|
case Type_Id:
|
|
|
|
case Type_IdRegEx:
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2015-03-28 11:05:49 +00:00
|
|
|
if (CSMWorld::ColumnBase::isId(display) || CSMWorld::ColumnBase::isScript(display))
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
|
|
|
consider = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
case Type_RecordState:
|
|
|
|
|
|
|
|
if (display == CSMWorld::ColumnBase::Display_RecordState)
|
|
|
|
consider = true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Type_None:
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (consider)
|
2015-03-28 11:05:49 +00:00
|
|
|
mColumns.insert(i);
|
2015-03-25 10:56:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mIdColumn = model->findColumnIndex(CSMWorld::Columns::ColumnId_Id);
|
|
|
|
mTypeColumn = model->findColumnIndex(CSMWorld::Columns::ColumnId_RecordType);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::Search::searchRow(const CSMWorld::IdTableBase* model, int row, CSMDoc::Messages& messages) const
|
|
|
|
{
|
2015-03-28 11:05:49 +00:00
|
|
|
for (std::set<int>::const_iterator iter(mColumns.begin()); iter != mColumns.end(); ++iter)
|
2015-03-25 10:56:14 +00:00
|
|
|
{
|
2015-03-28 11:05:49 +00:00
|
|
|
QModelIndex index = model->index(row, *iter);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
CSMWorld::UniversalId::Type type
|
|
|
|
= static_cast<CSMWorld::UniversalId::Type>(model->data(model->index(row, mTypeColumn)).toInt());
|
|
|
|
|
|
|
|
CSMWorld::UniversalId id(type, model->data(model->index(row, mIdColumn)).toString().toUtf8().data());
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
bool writable = model->flags(index) & Qt::ItemIsEditable;
|
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
switch (mType)
|
|
|
|
{
|
|
|
|
case Type_Text:
|
2015-03-28 10:54:32 +00:00
|
|
|
case Type_Id:
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
searchTextCell(model, index, id, writable, messages);
|
2015-03-25 10:56:14 +00:00
|
|
|
break;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
case Type_TextRegEx:
|
2015-03-28 10:54:32 +00:00
|
|
|
case Type_IdRegEx:
|
2015-03-25 10:56:14 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
searchRegExCell(model, index, id, writable, messages);
|
2015-03-25 10:56:14 +00:00
|
|
|
break;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-03-25 10:56:14 +00:00
|
|
|
case Type_RecordState:
|
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
searchRecordStateCell(model, index, id, writable, messages);
|
2015-03-25 10:56:14 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case Type_None:
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-03-30 20:30:33 +00:00
|
|
|
|
|
|
|
void CSMTools::Search::setPadding(int before, int after)
|
|
|
|
{
|
|
|
|
mPaddingBefore = before;
|
|
|
|
mPaddingAfter = after;
|
|
|
|
}
|
2015-04-16 16:50:22 +00:00
|
|
|
|
|
|
|
void CSMTools::Search::replace(CSMDoc::Document& document, CSMWorld::IdTableBase* model,
|
|
|
|
const CSMWorld::UniversalId& id, const std::string& messageHint, const std::string& replaceText) const
|
|
|
|
{
|
|
|
|
std::istringstream stream(messageHint.c_str());
|
|
|
|
|
|
|
|
char hint, ignore;
|
|
|
|
int columnId, pos, length;
|
|
|
|
|
|
|
|
if (stream >> hint >> ignore >> columnId >> pos >> length)
|
|
|
|
{
|
|
|
|
int column = model->findColumnIndex(static_cast<CSMWorld::Columns::ColumnId>(columnId));
|
|
|
|
|
|
|
|
QModelIndex index = model->getModelIndex(id.getId(), column);
|
|
|
|
|
|
|
|
std::string text = model->data(index).toString().toUtf8().constData();
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
std::string before = text.substr(0, pos);
|
|
|
|
std::string after = text.substr(pos + length);
|
|
|
|
|
|
|
|
std::string newText = before + replaceText + after;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-04-16 16:50:22 +00:00
|
|
|
document.getUndoStack().push(new CSMWorld::ModifyCommand(*model, index, QString::fromUtf8(newText.c_str())));
|
|
|
|
}
|
|
|
|
}
|
2015-04-27 20:43:09 +00:00
|
|
|
|
|
|
|
bool CSMTools::Search::verify(CSMDoc::Document& document, CSMWorld::IdTableBase* model, const CSMWorld::UniversalId& id,
|
|
|
|
const std::string& messageHint) const
|
|
|
|
{
|
2015-06-20 17:04:19 +00:00
|
|
|
CSMDoc::Messages messages(CSMDoc::Message::Severity_Info);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-04-27 20:43:09 +00:00
|
|
|
int row = model->getModelIndex(id.getId(), model->findColumnIndex(CSMWorld::Columns::ColumnId_Id)).row();
|
|
|
|
|
|
|
|
searchRow(model, row, messages);
|
|
|
|
|
|
|
|
for (CSMDoc::Messages::Iterator iter(messages.begin()); iter != messages.end(); ++iter)
|
|
|
|
if (iter->mHint == messageHint)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|