added replace function

test
Marc Zinnschlag 10 years ago
parent 3f4f008c51
commit 4951fc477c

@ -80,7 +80,7 @@ QVariant CSMTools::ReportModel::data (const QModelIndex & index, int role) const
char type, ignore; char type, ignore;
int fieldIndex; int fieldIndex;
if ((stream >> type >> ignore >> fieldIndex) && type=='r') if ((stream >> type >> ignore >> fieldIndex) && (type=='r' || type=='R'))
{ {
field = CSMWorld::Columns::getName ( field = CSMWorld::Columns::getName (
static_cast<CSMWorld::Columns::ColumnId> (fieldIndex)); static_cast<CSMWorld::Columns::ColumnId> (fieldIndex));
@ -135,6 +135,21 @@ void CSMTools::ReportModel::add (const CSMWorld::UniversalId& id, const std::str
endInsertRows(); endInsertRows();
} }
void CSMTools::ReportModel::flagAsReplaced (int index)
{
Line& line = mRows.at (index);
std::string hint = line.mHint;
if (hint.empty() || hint[0]!='R')
throw std::logic_error ("trying to flag message as replaced that is not replaceable");
hint[0] = 'r';
line.mHint = hint;
emit dataChanged (this->index (index, 0), this->index (index, columnCount()));
}
const CSMWorld::UniversalId& CSMTools::ReportModel::getUniversalId (int row) const const CSMWorld::UniversalId& CSMTools::ReportModel::getUniversalId (int row) const
{ {
return mRows.at (row).mId; return mRows.at (row).mId;

@ -53,6 +53,8 @@ namespace CSMTools
void add (const CSMWorld::UniversalId& id, const std::string& message, void add (const CSMWorld::UniversalId& id, const std::string& message,
const std::string& hint = ""); const std::string& hint = "");
void flagAsReplaced (int index);
const CSMWorld::UniversalId& getUniversalId (int row) const; const CSMWorld::UniversalId& getUniversalId (int row) const;
std::string getHint (int row) const; std::string getHint (int row) const;

@ -4,14 +4,16 @@
#include <stdexcept> #include <stdexcept>
#include <sstream> #include <sstream>
#include "../../model/doc/messages.hpp" #include "../doc/messages.hpp"
#include "../doc/document.hpp"
#include "../../model/world/idtablebase.hpp" #include "../world/idtablebase.hpp"
#include "../../model/world/columnbase.hpp" #include "../world/columnbase.hpp"
#include "../../model/world/universalid.hpp" #include "../world/universalid.hpp"
#include "../world/commands.hpp"
void CSMTools::Search::searchTextCell (const CSMWorld::IdTableBase *model, void CSMTools::Search::searchTextCell (const CSMWorld::IdTableBase *model,
const QModelIndex& index, const CSMWorld::UniversalId& id, const QModelIndex& index, const CSMWorld::UniversalId& id, bool writable,
CSMDoc::Messages& messages) const CSMDoc::Messages& messages) const
{ {
// using QString here for easier handling of case folding. // using QString here for easier handling of case folding.
@ -25,7 +27,8 @@ void CSMTools::Search::searchTextCell (const CSMWorld::IdTableBase *model,
{ {
std::ostringstream hint; std::ostringstream hint;
hint hint
<< "r: " << (writable ? 'R' : 'r')
<<": "
<< model->getColumnId (index.column()) << model->getColumnId (index.column())
<< " " << pos << " " << pos
<< " " << search.length(); << " " << search.length();
@ -37,7 +40,7 @@ void CSMTools::Search::searchTextCell (const CSMWorld::IdTableBase *model,
} }
void CSMTools::Search::searchRegExCell (const CSMWorld::IdTableBase *model, void CSMTools::Search::searchRegExCell (const CSMWorld::IdTableBase *model,
const QModelIndex& index, const CSMWorld::UniversalId& id, const QModelIndex& index, const CSMWorld::UniversalId& id, bool writable,
CSMDoc::Messages& messages) const CSMDoc::Messages& messages) const
{ {
QString text = model->data (index).toString(); QString text = model->data (index).toString();
@ -49,7 +52,12 @@ void CSMTools::Search::searchRegExCell (const CSMWorld::IdTableBase *model,
int length = mRegExp.matchedLength(); int length = mRegExp.matchedLength();
std::ostringstream hint; std::ostringstream hint;
hint << "r: " << model->getColumnId (index.column()) << " " << pos << " " << length; hint
<< (writable ? 'R' : 'r')
<<": "
<< model->getColumnId (index.column())
<< " " << pos
<< " " << length;
messages.add (id, formatDescription (text, pos, length).toUtf8().data(), hint.str()); messages.add (id, formatDescription (text, pos, length).toUtf8().data(), hint.str());
@ -58,8 +66,11 @@ void CSMTools::Search::searchRegExCell (const CSMWorld::IdTableBase *model,
} }
void CSMTools::Search::searchRecordStateCell (const CSMWorld::IdTableBase *model, void CSMTools::Search::searchRecordStateCell (const CSMWorld::IdTableBase *model,
const QModelIndex& index, const CSMWorld::UniversalId& id, CSMDoc::Messages& messages) const const QModelIndex& index, const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const
{ {
if (writable)
throw std::logic_error ("Record state can not be modified by search and replace");
int data = model->data (index).toInt(); int data = model->data (index).toInt();
if (data==mValue) if (data==mValue)
@ -204,23 +215,25 @@ void CSMTools::Search::searchRow (const CSMWorld::IdTableBase *model, int row,
CSMWorld::UniversalId id ( CSMWorld::UniversalId id (
type, model->data (model->index (row, mIdColumn)).toString().toUtf8().data()); type, model->data (model->index (row, mIdColumn)).toString().toUtf8().data());
bool writable = model->flags (index) & Qt::ItemIsEditable;
switch (mType) switch (mType)
{ {
case Type_Text: case Type_Text:
case Type_Id: case Type_Id:
searchTextCell (model, index, id, messages); searchTextCell (model, index, id, writable, messages);
break; break;
case Type_TextRegEx: case Type_TextRegEx:
case Type_IdRegEx: case Type_IdRegEx:
searchRegExCell (model, index, id, messages); searchRegExCell (model, index, id, writable, messages);
break; break;
case Type_RecordState: case Type_RecordState:
searchRecordStateCell (model, index, id, messages); searchRecordStateCell (model, index, id, writable, messages);
break; break;
case Type_None: case Type_None:
@ -235,3 +248,32 @@ void CSMTools::Search::setPadding (int before, int after)
mPaddingBefore = before; mPaddingBefore = before;
mPaddingAfter = after; mPaddingAfter = after;
} }
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();
std::string before = text.substr (0, pos);
std::string after = text.substr (pos+length);
std::string newText = before + replaceText + after;
document.getUndoStack().push (
new CSMWorld::ModifyCommand (*model, index, QString::fromUtf8 (newText.c_str())));
}
}

@ -12,6 +12,7 @@ class QModelIndex;
namespace CSMDoc namespace CSMDoc
{ {
class Messages; class Messages;
class Document;
} }
namespace CSMWorld namespace CSMWorld
@ -49,13 +50,13 @@ namespace CSMTools
int mPaddingAfter; int mPaddingAfter;
void searchTextCell (const CSMWorld::IdTableBase *model, const QModelIndex& index, void searchTextCell (const CSMWorld::IdTableBase *model, const QModelIndex& index,
const CSMWorld::UniversalId& id, CSMDoc::Messages& messages) const; const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const;
void searchRegExCell (const CSMWorld::IdTableBase *model, const QModelIndex& index, void searchRegExCell (const CSMWorld::IdTableBase *model, const QModelIndex& index,
const CSMWorld::UniversalId& id, CSMDoc::Messages& messages) const; const CSMWorld::UniversalId& id, bool writable, CSMDoc::Messages& messages) const;
void searchRecordStateCell (const CSMWorld::IdTableBase *model, void searchRecordStateCell (const CSMWorld::IdTableBase *model,
const QModelIndex& index, const CSMWorld::UniversalId& id, const QModelIndex& index, const CSMWorld::UniversalId& id, bool writable,
CSMDoc::Messages& messages) const; CSMDoc::Messages& messages) const;
QString formatDescription (const QString& description, int pos, int length) const; QString formatDescription (const QString& description, int pos, int length) const;
@ -82,6 +83,11 @@ namespace CSMTools
CSMDoc::Messages& messages) const; CSMDoc::Messages& messages) const;
void setPadding (int before, int after); void setPadding (int before, int after);
// Configuring *this for the model is not necessary when calling this function.
void replace (CSMDoc::Document& document, CSMWorld::IdTableBase *model,
const CSMWorld::UniversalId& id, const std::string& messageHint,
const std::string& replaceText) const;
}; };
} }

@ -56,6 +56,23 @@ void CSVTools::ReportTable::contextMenuEvent (QContextMenuEvent *event)
{ {
menu.addAction (mShowAction); menu.addAction (mShowAction);
menu.addAction (mRemoveAction); menu.addAction (mRemoveAction);
bool found = false;
for (QModelIndexList::const_iterator iter (selectedRows.begin());
iter!=selectedRows.end(); ++iter)
{
QString hint = mModel->data (mModel->index (iter->row(), 2)).toString();
if (!hint.isEmpty() && hint[0]=='R')
{
found = true;
break;
}
}
if (found)
menu.addAction (mReplaceAction);
} }
menu.exec (event->globalPos()); menu.exec (event->globalPos());
@ -129,6 +146,10 @@ CSVTools::ReportTable::ReportTable (CSMDoc::Document& document,
mRemoveAction = new QAction (tr ("Remove from list"), this); mRemoveAction = new QAction (tr ("Remove from list"), this);
connect (mRemoveAction, SIGNAL (triggered()), this, SLOT (removeSelection())); connect (mRemoveAction, SIGNAL (triggered()), this, SLOT (removeSelection()));
addAction (mRemoveAction); addAction (mRemoveAction);
mReplaceAction = new QAction (tr ("Replace"), this);
connect (mReplaceAction, SIGNAL (triggered()), this, SIGNAL (replaceRequest()));
addAction (mReplaceAction);
} }
std::vector<CSMWorld::UniversalId> CSVTools::ReportTable::getDraggedRecords() const std::vector<CSMWorld::UniversalId> CSVTools::ReportTable::getDraggedRecords() const
@ -151,6 +172,42 @@ void CSVTools::ReportTable::updateUserSetting (const QString& name, const QStrin
mIdTypeDelegate->updateUserSetting (name, list); mIdTypeDelegate->updateUserSetting (name, list);
} }
std::vector<int> CSVTools::ReportTable::getReplaceIndices (bool selection) const
{
std::vector<int> indices;
if (selection)
{
QModelIndexList selectedRows = selectionModel()->selectedRows();
for (QModelIndexList::const_iterator iter (selectedRows.begin()); iter!=selectedRows.end();
++iter)
{
QString hint = mModel->data (mModel->index (iter->row(), 2)).toString();
if (!hint.isEmpty() && hint[0]=='R')
indices.push_back (iter->row());
}
}
else
{
for (int i=0; i<mModel->rowCount(); ++i)
{
QString hint = mModel->data (mModel->index (i, 2)).toString();
if (!hint.isEmpty() && hint[0]=='R')
indices.push_back (i);
}
}
return indices;
}
void CSVTools::ReportTable::flagAsReplaced (int index)
{
mModel->flagAsReplaced (index);
}
void CSVTools::ReportTable::showSelection() void CSVTools::ReportTable::showSelection()
{ {
QModelIndexList selectedRows = selectionModel()->selectedRows(); QModelIndexList selectedRows = selectionModel()->selectedRows();

@ -25,6 +25,7 @@ namespace CSVTools
CSVWorld::CommandDelegate *mIdTypeDelegate; CSVWorld::CommandDelegate *mIdTypeDelegate;
QAction *mShowAction; QAction *mShowAction;
QAction *mRemoveAction; QAction *mRemoveAction;
QAction *mReplaceAction;
private: private:
@ -46,6 +47,13 @@ namespace CSVTools
void clear(); void clear();
// Return indices of rows that are suitable for replacement.
//
// \param selection Only list selected rows.
std::vector<int> getReplaceIndices (bool selection) const;
void flagAsReplaced (int index);
private slots: private slots:
void showSelection(); void showSelection();
@ -55,6 +63,8 @@ namespace CSVTools
signals: signals:
void editRequest (const CSMWorld::UniversalId& id, const std::string& hint); void editRequest (const CSMWorld::UniversalId& id, const std::string& hint);
void replaceRequest();
}; };
} }

@ -38,6 +38,9 @@ void CSVTools::SearchBox::updateSearchButton()
CSVTools::SearchBox::SearchBox (QWidget *parent) CSVTools::SearchBox::SearchBox (QWidget *parent)
: QWidget (parent), mSearch ("Search"), mSearchEnabled (false) : QWidget (parent), mSearch ("Search"), mSearchEnabled (false)
{ {
mLayout = new QGridLayout (this);
// search panel
std::vector<std::string> states = std::vector<std::string> states =
CSMWorld::Columns::getEnums (CSMWorld::Columns::ColumnId_Modification); CSMWorld::Columns::getEnums (CSMWorld::Columns::ColumnId_Modification);
states.resize (states.size()-1); // ignore erased state states.resize (states.size()-1); // ignore erased state
@ -46,8 +49,6 @@ CSVTools::SearchBox::SearchBox (QWidget *parent)
++iter) ++iter)
mRecordState.addItem (QString::fromUtf8 (iter->c_str())); mRecordState.addItem (QString::fromUtf8 (iter->c_str()));
mLayout = new QGridLayout (this);
mMode.addItem ("Text"); mMode.addItem ("Text");
mMode.addItem ("Text (RegEx)"); mMode.addItem ("Text (RegEx)");
mMode.addItem ("ID"); mMode.addItem ("ID");
@ -63,11 +64,6 @@ CSVTools::SearchBox::SearchBox (QWidget *parent)
mLayout->addWidget (&mInput, 0, 1); mLayout->addWidget (&mInput, 0, 1);
mLayout->setColumnMinimumWidth (2, 50);
mLayout->setColumnStretch (1, 1);
mLayout->setContentsMargins (0, 0, 0, 0);
connect (&mMode, SIGNAL (activated (int)), this, SLOT (modeSelected (int))); connect (&mMode, SIGNAL (activated (int)), this, SLOT (modeSelected (int)));
connect (&mText, SIGNAL (textChanged (const QString&)), connect (&mText, SIGNAL (textChanged (const QString&)),
@ -77,6 +73,19 @@ CSVTools::SearchBox::SearchBox (QWidget *parent)
connect (&mText, SIGNAL (returnPressed()), this, SLOT (startSearch())); connect (&mText, SIGNAL (returnPressed()), this, SLOT (startSearch()));
// replace panel
mReplaceInput.insertWidget (0, &mReplaceText);
mReplaceInput.insertWidget (1, &mReplacePlaceholder);
mLayout->addWidget (&mReplaceInput, 1, 1);
// layout adjustments
mLayout->setColumnMinimumWidth (2, 50);
mLayout->setColumnStretch (1, 1);
mLayout->setContentsMargins (0, 0, 0, 0);
// update
modeSelected (0); modeSelected (0);
updateSearchButton(); updateSearchButton();
@ -116,6 +125,25 @@ CSMTools::Search CSVTools::SearchBox::getSearch() const
throw std::logic_error ("invalid search mode index"); throw std::logic_error ("invalid search mode index");
} }
std::string CSVTools::SearchBox::getReplaceText() const
{
CSMTools::Search::Type type = static_cast<CSMTools::Search::Type> (mMode.currentIndex());
switch (type)
{
case CSMTools::Search::Type_Text:
case CSMTools::Search::Type_TextRegEx:
case CSMTools::Search::Type_Id:
case CSMTools::Search::Type_IdRegEx:
return mReplaceText.text().toUtf8().data();
default:
throw std::logic_error ("Invalid search mode for replace");
}
}
void CSVTools::SearchBox::modeSelected (int index) void CSVTools::SearchBox::modeSelected (int index)
{ {
switch (index) switch (index)
@ -126,10 +154,12 @@ void CSVTools::SearchBox::modeSelected (int index)
case CSMTools::Search::Type_IdRegEx: case CSMTools::Search::Type_IdRegEx:
mInput.setCurrentIndex (0); mInput.setCurrentIndex (0);
mReplaceInput.setCurrentIndex (0);
break; break;
case CSMTools::Search::Type_RecordState: case CSMTools::Search::Type_RecordState:
mInput.setCurrentIndex (1); mInput.setCurrentIndex (1);
mReplaceInput.setCurrentIndex (1);
break; break;
} }

@ -6,6 +6,7 @@
#include <QComboBox> #include <QComboBox>
#include <QStackedWidget> #include <QStackedWidget>
#include <QPushButton> #include <QPushButton>
#include <QLabel>
class QGridLayout; class QGridLayout;
@ -27,6 +28,9 @@ namespace CSVTools
QGridLayout *mLayout; QGridLayout *mLayout;
QComboBox mMode; QComboBox mMode;
bool mSearchEnabled; bool mSearchEnabled;
QStackedWidget mReplaceInput;
QLineEdit mReplaceText;
QLabel mReplacePlaceholder;
private: private:
@ -40,6 +44,8 @@ namespace CSVTools
CSMTools::Search getSearch() const; CSMTools::Search getSearch() const;
std::string getReplaceText() const;
private slots: private slots:
void modeSelected (int index); void modeSelected (int index);

@ -5,6 +5,8 @@
#include "../../model/doc/document.hpp" #include "../../model/doc/document.hpp"
#include "../../model/tools/search.hpp" #include "../../model/tools/search.hpp"
#include "../../model/tools/reportmodel.hpp"
#include "../../model/world/idtablebase.hpp"
#include "reporttable.hpp" #include "reporttable.hpp"
#include "searchbox.hpp" #include "searchbox.hpp"
@ -31,6 +33,8 @@ CSVTools::SearchSubView::SearchSubView (const CSMWorld::UniversalId& id, CSMDoc:
connect (mTable, SIGNAL (editRequest (const CSMWorld::UniversalId&, const std::string&)), connect (mTable, SIGNAL (editRequest (const CSMWorld::UniversalId&, const std::string&)),
SIGNAL (focusId (const CSMWorld::UniversalId&, const std::string&))); SIGNAL (focusId (const CSMWorld::UniversalId&, const std::string&)));
connect (mTable, SIGNAL (replaceRequest()), this, SLOT (replaceRequest()));
connect (&document, SIGNAL (stateChanged (int, CSMDoc::Document *)), connect (&document, SIGNAL (stateChanged (int, CSMDoc::Document *)),
this, SLOT (stateChanged (int, CSMDoc::Document *))); this, SLOT (stateChanged (int, CSMDoc::Document *)));
@ -63,9 +67,36 @@ void CSVTools::SearchSubView::stateChanged (int state, CSMDoc::Document *documen
void CSVTools::SearchSubView::startSearch (const CSMTools::Search& search) void CSVTools::SearchSubView::startSearch (const CSMTools::Search& search)
{ {
CSMTools::Search search2 (search); mSearch = search;
search2.setPadding (mPaddingBefore, mPaddingAfter); mSearch.setPadding (mPaddingBefore, mPaddingAfter);
mTable->clear(); mTable->clear();
mDocument.runSearch (getUniversalId(), search2); mDocument.runSearch (getUniversalId(), mSearch);
}
void CSVTools::SearchSubView::replaceRequest()
{
std::vector<int> indices = mTable->getReplaceIndices (true);
std::string replace = mSearchBox.getReplaceText();
const CSMTools::ReportModel& model =
dynamic_cast<const CSMTools::ReportModel&> (*mTable->model());
// We are running through the indices in reverse order to avoid messing up multiple results
// in a single string.
for (std::vector<int>::const_reverse_iterator iter (indices.rbegin()); iter!=indices.rend(); ++iter)
{
CSMWorld::UniversalId id = model.getUniversalId (*iter);
CSMWorld::UniversalId::Type type = CSMWorld::UniversalId::getParentType (id.getType());
CSMWorld::IdTableBase *table = &dynamic_cast<CSMWorld::IdTableBase&> (
*mDocument.getData().getTableModel (type));
std::string hint = model.getHint (*iter);
mSearch.replace (mDocument, table, id, hint, replace);
mTable->flagAsReplaced (*iter);
}
} }

@ -1,6 +1,8 @@
#ifndef CSV_TOOLS_SEARCHSUBVIEW_H #ifndef CSV_TOOLS_SEARCHSUBVIEW_H
#define CSV_TOOLS_SEARCHSUBVIEW_H #define CSV_TOOLS_SEARCHSUBVIEW_H
#include "../../model/tools/search.hpp"
#include "../doc/subview.hpp" #include "../doc/subview.hpp"
#include "searchbox.hpp" #include "searchbox.hpp"
@ -26,6 +28,7 @@ namespace CSVTools
CSMDoc::Document& mDocument; CSMDoc::Document& mDocument;
int mPaddingBefore; int mPaddingBefore;
int mPaddingAfter; int mPaddingAfter;
CSMTools::Search mSearch;
public: public:
@ -40,6 +43,8 @@ namespace CSVTools
void stateChanged (int state, CSMDoc::Document *document); void stateChanged (int state, CSMDoc::Document *document);
void startSearch (const CSMTools::Search& search); void startSearch (const CSMTools::Search& search);
void replaceRequest();
}; };
} }

Loading…
Cancel
Save