#include "search.hpp" #include #include #include "../../model/doc/messages.hpp" #include "../../model/world/idtablebase.hpp" #include "../../model/world/columnbase.hpp" #include "../../model/world/universalid.hpp" void CSMTools::Search::searchTextCell (const CSMWorld::IdTableBase *model, const QModelIndex& index, const CSMWorld::UniversalId& id, bool multiple, CSMDoc::Messages& messages) const { // using QString here for easier handling of case folding. QString search = QString::fromUtf8 (mText.c_str()); QString text = model->data (index).toString(); int pos = 0; while ((pos = text.indexOf (search, pos, Qt::CaseInsensitive))!=-1) { std::ostringstream message; message << text.mid (pos).toUtf8().data(); std::ostringstream hint; message << "r: " << index.column() << " " << pos << " " << search.length(); messages.add (id, message.str(), hint.str()); if (!multiple) break; pos += search.length(); } } void CSMTools::Search::searchRegExCell (const CSMWorld::IdTableBase *model, const QModelIndex& index, const CSMWorld::UniversalId& id, bool multiple, CSMDoc::Messages& messages) const { QString text = model->data (index).toString(); int pos = 0; while ((pos = mRegExp.indexIn (text, pos))!=-1) { std::ostringstream message; message << text.mid (pos).toUtf8().data(); int length = mRegExp.matchedLength(); std::ostringstream hint; message << "r: " << index.column() << " " << pos << " " << length; messages.add (id, message.str(), hint.str()); if (!multiple) break; pos += length; } } void CSMTools::Search::searchRecordStateCell (const CSMWorld::IdTableBase *model, const QModelIndex& index, const CSMWorld::UniversalId& id, CSMDoc::Messages& messages) const { int data = model->data (index).toInt(); if (data==mValue) { std::vector states = CSMWorld::Columns::getEnums (CSMWorld::Columns::ColumnId_Modification); std::ostringstream message; message << id.getId() << " " << states.at (data); std::ostringstream hint; message << "r: " << index.column(); messages.add (id, message.str(), hint.str()); } } CSMTools::Search::Search() : mType (Type_None) {} CSMTools::Search::Search (Type type, const std::string& value) : mType (type), mText (value) { if (type!=Type_Text && type!=Type_Reference) throw std::logic_error ("Invalid search parameter (string)"); } CSMTools::Search::Search (Type type, const QRegExp& value) : mType (type), mRegExp (value) { if (type!=Type_TextRegEx && type!=Type_ReferenceRegEx) throw std::logic_error ("Invalid search parameter (RegExp)"); } CSMTools::Search::Search (Type type, int value) : mType (type), mValue (value) { 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 ( model->headerData ( i, Qt::Horizontal, static_cast (CSMWorld::ColumnBase::Role_Display)).toInt()); bool consider = false; bool multiple = false; switch (mType) { case Type_Text: case Type_TextRegEx: if (CSMWorld::ColumnBase::isText (display) || CSMWorld::ColumnBase::isScript (display)) { consider = true; multiple = true; } break; case Type_Reference: case Type_ReferenceRegEx: if (CSMWorld::ColumnBase::isId (display)) { consider = true; } else if (CSMWorld::ColumnBase::isScript (display)) { consider = true; multiple = true; } break; case Type_RecordState: if (display==CSMWorld::ColumnBase::Display_RecordState) consider = true; break; case Type_None: break; } if (consider) mColumns.insert (std::make_pair (i, multiple)); } 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 { for (std::map::const_iterator iter (mColumns.begin()); iter!=mColumns.end(); ++iter) { QModelIndex index = model->index (row, iter->first); CSMWorld::UniversalId::Type type = static_cast ( model->data (model->index (row, mTypeColumn)).toInt()); CSMWorld::UniversalId id ( type, model->data (model->index (row, mIdColumn)).toString().toUtf8().data()); switch (mType) { case Type_Text: case Type_Reference: searchTextCell (model, index, id, iter->second, messages); break; case Type_TextRegEx: case Type_ReferenceRegEx: searchRegExCell (model, index, id, iter->second, messages); break; case Type_RecordState: searchRecordStateCell (model, index, id, messages); break; case Type_None: break; } } }