2013-10-29 12:18:22 +00:00
|
|
|
#include "infocollection.hpp"
|
|
|
|
|
2013-11-08 11:16:41 +00:00
|
|
|
#include <stdexcept>
|
2013-11-09 10:42:19 +00:00
|
|
|
#include <iterator>
|
2021-08-01 04:16:37 +00:00
|
|
|
#include <cassert>
|
2013-11-08 11:16:41 +00:00
|
|
|
|
2022-01-22 14:58:41 +00:00
|
|
|
#include <components/esm3/esmreader.hpp>
|
|
|
|
#include <components/esm3/loaddial.hpp>
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2022-08-02 22:00:54 +00:00
|
|
|
#include <components/misc/strings/algorithm.hpp>
|
2013-11-08 10:52:30 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
namespace CSMWorld
|
2013-10-29 12:18:22 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
template<>
|
|
|
|
void Collection<Info, IdAccessor<Info> >::removeRows (int index, int count)
|
|
|
|
{
|
|
|
|
mRecords.erase(mRecords.begin()+index, mRecords.begin()+index+count);
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// index map is updated in InfoCollection::removeRows()
|
|
|
|
}
|
|
|
|
|
|
|
|
template<>
|
|
|
|
void Collection<Info, IdAccessor<Info> >::insertRecord (std::unique_ptr<RecordBase> record,
|
|
|
|
int index, UniversalId::Type type)
|
2013-10-29 12:18:22 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
int size = static_cast<int>(mRecords.size());
|
|
|
|
if (index < 0 || index > size)
|
|
|
|
throw std::runtime_error("index out of range");
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
std::unique_ptr<Record<Info> > record2(static_cast<Record<Info>*>(record.release()));
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
if (index == size)
|
|
|
|
mRecords.push_back(std::move(record2));
|
|
|
|
else
|
|
|
|
mRecords.insert(mRecords.begin()+index, std::move(record2));
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// index map is updated in InfoCollection::insertRecord()
|
|
|
|
}
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
template<>
|
|
|
|
bool Collection<Info, IdAccessor<Info> >::reorderRowsImp (int baseIndex,
|
|
|
|
const std::vector<int>& newOrder)
|
|
|
|
{
|
|
|
|
if (!newOrder.empty())
|
2013-11-09 10:42:19 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
int size = static_cast<int>(newOrder.size());
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// check that all indices are present
|
|
|
|
std::vector<int> test(newOrder);
|
|
|
|
std::sort(test.begin(), test.end());
|
|
|
|
if (*test.begin() != 0 || *--test.end() != size-1)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// reorder records
|
|
|
|
std::vector<std::unique_ptr<Record<Info> > > buffer(size);
|
|
|
|
|
|
|
|
// FIXME: BUG: undo does not remove modified flag
|
|
|
|
for (int i = 0; i < size; ++i)
|
|
|
|
{
|
|
|
|
buffer[newOrder[i]] = std::move(mRecords[baseIndex+i]);
|
2022-07-12 08:56:53 +00:00
|
|
|
if (buffer[newOrder[i]])
|
|
|
|
buffer[newOrder[i]]->setModified(buffer[newOrder[i]]->get());
|
2021-07-23 06:05:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::move(buffer.begin(), buffer.end(), mRecords.begin()+baseIndex);
|
2013-11-10 11:09:49 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// index map is updated in InfoCollection::reorderRows()
|
2013-11-10 11:09:49 +00:00
|
|
|
}
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMWorld::InfoCollection::load (const Info& record, bool base)
|
|
|
|
{
|
|
|
|
int index = searchId (record.mId);
|
|
|
|
|
|
|
|
if (index==-1)
|
|
|
|
{
|
|
|
|
// new record
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record2 = std::make_unique<Record<Info>>();
|
2021-07-23 06:05:58 +00:00
|
|
|
record2->mState = base ? RecordBase::State_BaseOnly : RecordBase::State_ModifiedOnly;
|
|
|
|
(base ? record2->mBase : record2->mModified) = record;
|
|
|
|
|
|
|
|
appendRecord(std::move(record2));
|
2013-10-29 12:18:22 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// old record
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record2 = std::make_unique<Record<Info>>(getRecord(index));
|
2013-10-29 12:18:22 +00:00
|
|
|
|
|
|
|
if (base)
|
2021-07-23 04:21:21 +00:00
|
|
|
record2->mBase = record;
|
2013-10-29 12:18:22 +00:00
|
|
|
else
|
2021-07-23 04:21:21 +00:00
|
|
|
record2->setModified (record);
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2021-07-23 04:21:21 +00:00
|
|
|
setRecord (index, std::move(record2));
|
2013-10-29 12:18:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-04 16:07:23 +00:00
|
|
|
int CSMWorld::InfoCollection::getInfoIndex(std::string_view id, std::string_view topic) const
|
2013-11-09 10:42:19 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
// find the topic first
|
2021-07-23 23:17:48 +00:00
|
|
|
std::unordered_map<std::string, std::vector<std::pair<std::string, int> > >::const_iterator iter
|
|
|
|
= mInfoIndex.find(Misc::StringUtils::lowerCase(topic));
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
if (iter == mInfoIndex.end())
|
|
|
|
return -1;
|
2013-11-09 10:42:19 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// brute force loop
|
|
|
|
for (std::vector<std::pair<std::string, int> >::const_iterator it = iter->second.begin();
|
|
|
|
it != iter->second.end(); ++it)
|
|
|
|
{
|
|
|
|
if (Misc::StringUtils::ciEqual(it->first, id))
|
|
|
|
return it->second;
|
|
|
|
}
|
2013-11-09 10:42:19 +00:00
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// Calling insertRecord() using index from getInsertIndex() needs to take into account of
|
|
|
|
// prev/next records; an example is deleting a record then undo
|
|
|
|
int CSMWorld::InfoCollection::getInsertIndex (const std::string& id,
|
|
|
|
UniversalId::Type type, RecordBase *record) const
|
2013-11-08 11:16:41 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
if (record == nullptr)
|
|
|
|
{
|
|
|
|
std::string::size_type separator = id.find_last_of('#');
|
2013-11-08 11:16:41 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
if (separator == std::string::npos)
|
|
|
|
throw std::runtime_error("invalid info ID: " + id);
|
2013-11-08 11:16:41 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
std::pair<RecordConstIterator, RecordConstIterator> range = getTopicRange(id.substr(0, separator));
|
2013-11-08 11:16:41 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
if (range.first == range.second)
|
|
|
|
return Collection<Info, IdAccessor<Info> >::getAppendIndex(id, type);
|
|
|
|
|
|
|
|
return std::distance(getRecords().begin(), range.second);
|
|
|
|
}
|
2013-11-08 11:16:41 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
int index = -1;
|
|
|
|
|
|
|
|
const Info& info = static_cast<Record<Info>*>(record)->get();
|
|
|
|
std::string topic = info.mTopicId;
|
|
|
|
|
|
|
|
// if the record has a prev, find its index value
|
|
|
|
if (!info.mPrev.empty())
|
|
|
|
{
|
|
|
|
index = getInfoIndex(info.mPrev, topic);
|
|
|
|
|
|
|
|
if (index != -1)
|
|
|
|
++index; // if prev exists, set current index to one above prev
|
|
|
|
}
|
|
|
|
|
|
|
|
// if prev doesn't exist or not found and the record has a next, find its index value
|
|
|
|
if (index == -1 && !info.mNext.empty())
|
|
|
|
{
|
|
|
|
// if next exists, use its index as the current index
|
|
|
|
index = getInfoIndex(info.mNext, topic);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if next doesn't exist or not found (i.e. neither exist yet) then start a new one
|
|
|
|
if (index == -1)
|
|
|
|
{
|
|
|
|
Range range = getTopicRange(topic); // getTopicRange converts topic to lower case first
|
|
|
|
|
|
|
|
index = std::distance(getRecords().begin(), range.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
return index;
|
2013-11-08 11:16:41 +00:00
|
|
|
}
|
|
|
|
|
2013-11-14 10:39:14 +00:00
|
|
|
bool CSMWorld::InfoCollection::reorderRows (int baseIndex, const std::vector<int>& newOrder)
|
|
|
|
{
|
|
|
|
// check if the range is valid
|
|
|
|
int lastIndex = baseIndex + newOrder.size() -1;
|
|
|
|
|
|
|
|
if (lastIndex>=getSize())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check that topics match
|
2015-06-24 17:01:29 +00:00
|
|
|
if (!Misc::StringUtils::ciEqual(getRecord(baseIndex).get().mTopicId,
|
|
|
|
getRecord(lastIndex).get().mTopicId))
|
2013-11-14 10:39:14 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// reorder
|
2021-07-23 06:05:58 +00:00
|
|
|
if (!Collection<Info, IdAccessor<Info> >::reorderRowsImp(baseIndex, newOrder))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// adjust index
|
|
|
|
int size = static_cast<int>(newOrder.size());
|
2021-07-23 23:17:48 +00:00
|
|
|
for (auto& [hash, infos] : mInfoIndex)
|
|
|
|
for (auto& [a, b] : infos)
|
|
|
|
if (b >= baseIndex && b < baseIndex + size)
|
|
|
|
b = newOrder.at(b - baseIndex) + baseIndex;
|
2021-07-23 06:05:58 +00:00
|
|
|
|
|
|
|
return true;
|
2013-11-14 10:39:14 +00:00
|
|
|
}
|
|
|
|
|
2013-10-31 11:16:45 +00:00
|
|
|
void CSMWorld::InfoCollection::load (ESM::ESMReader& reader, bool base, const ESM::Dialogue& dialogue)
|
2013-10-29 12:18:22 +00:00
|
|
|
{
|
2015-07-13 08:19:14 +00:00
|
|
|
Info info;
|
2015-07-21 17:25:43 +00:00
|
|
|
bool isDeleted = false;
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2015-07-21 17:25:43 +00:00
|
|
|
info.load (reader, isDeleted);
|
2015-07-13 08:19:14 +00:00
|
|
|
std::string id = Misc::StringUtils::lowerCase (dialogue.mId) + "#" + info.mId;
|
2013-10-29 12:18:22 +00:00
|
|
|
|
2015-07-21 17:25:43 +00:00
|
|
|
if (isDeleted)
|
2013-10-29 12:18:22 +00:00
|
|
|
{
|
|
|
|
int index = searchId (id);
|
|
|
|
|
|
|
|
if (index==-1)
|
|
|
|
{
|
|
|
|
// deleting a record that does not exist
|
|
|
|
// ignore it for now
|
|
|
|
/// \todo report the problem to the user
|
|
|
|
}
|
|
|
|
else if (base)
|
|
|
|
{
|
|
|
|
removeRows (index, 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record = std::make_unique<Record<Info>>(getRecord(index));
|
2021-07-23 04:21:21 +00:00
|
|
|
record->mState = RecordBase::State_Deleted;
|
|
|
|
setRecord (index, std::move(record));
|
2013-10-29 12:18:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-07-13 08:19:14 +00:00
|
|
|
info.mTopicId = dialogue.mId;
|
|
|
|
info.mId = id;
|
|
|
|
load (info, base);
|
2013-10-29 12:18:22 +00:00
|
|
|
}
|
|
|
|
}
|
2013-11-08 10:52:30 +00:00
|
|
|
|
2013-11-10 12:00:46 +00:00
|
|
|
CSMWorld::InfoCollection::Range CSMWorld::InfoCollection::getTopicRange (const std::string& topic)
|
|
|
|
const
|
2013-11-08 10:52:30 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
std::string lowerTopic = Misc::StringUtils::lowerCase (topic);
|
2013-11-08 10:52:30 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// find the topic
|
2021-07-23 23:17:48 +00:00
|
|
|
std::unordered_map<std::string, std::vector<std::pair<std::string, int> > >::const_iterator iter
|
|
|
|
= mInfoIndex.find(lowerTopic);
|
2013-11-10 12:00:46 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
if (iter == mInfoIndex.end())
|
2013-11-10 12:00:46 +00:00
|
|
|
return Range (getRecords().end(), getRecords().end());
|
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// topic found, find the starting index
|
|
|
|
int low = INT_MAX;
|
|
|
|
for (std::vector<std::pair<std::string, int> >::const_iterator it = iter->second.begin();
|
|
|
|
it != iter->second.end(); ++it)
|
2015-05-26 07:38:22 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
low = std::min(low, it->second);
|
2015-05-26 07:38:22 +00:00
|
|
|
}
|
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
RecordConstIterator begin = getRecords().begin() + low;
|
2013-11-08 10:52:30 +00:00
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
// Find end (one past the range)
|
|
|
|
RecordConstIterator end = begin + iter->second.size();
|
2021-08-01 04:16:37 +00:00
|
|
|
|
|
|
|
assert(static_cast<size_t>(std::distance(begin, end)) == iter->second.size());
|
2013-11-08 10:52:30 +00:00
|
|
|
|
2013-11-10 12:00:46 +00:00
|
|
|
return Range (begin, end);
|
2014-01-14 08:46:53 +00:00
|
|
|
}
|
2015-07-18 17:32:10 +00:00
|
|
|
|
|
|
|
void CSMWorld::InfoCollection::removeDialogueInfos(const std::string& dialogueId)
|
|
|
|
{
|
|
|
|
std::vector<int> erasedRecords;
|
|
|
|
|
2021-07-23 06:05:58 +00:00
|
|
|
Range range = getTopicRange(dialogueId); // getTopicRange converts dialogueId to lower case first
|
|
|
|
|
|
|
|
for (; range.first != range.second; ++range.first)
|
2015-07-18 17:32:10 +00:00
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
const Record<Info>& record = **range.first;
|
2015-07-18 17:32:10 +00:00
|
|
|
|
|
|
|
if (Misc::StringUtils::ciEqual(dialogueId, record.get().mTopicId))
|
|
|
|
{
|
|
|
|
if (record.mState == RecordBase::State_ModifiedOnly)
|
|
|
|
{
|
2021-07-23 06:05:58 +00:00
|
|
|
erasedRecords.push_back(range.first - getRecords().begin());
|
2015-07-18 17:32:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record2 = std::make_unique<Record<Info>>(record);
|
2021-07-23 04:21:21 +00:00
|
|
|
record2->mState = RecordBase::State_Deleted;
|
2021-07-23 06:05:58 +00:00
|
|
|
setRecord(range.first - getRecords().begin(), std::move(record2));
|
2015-07-18 17:32:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!erasedRecords.empty())
|
|
|
|
{
|
|
|
|
removeRows(erasedRecords.back(), 1);
|
|
|
|
erasedRecords.pop_back();
|
|
|
|
}
|
|
|
|
}
|
2021-07-23 06:05:58 +00:00
|
|
|
|
|
|
|
// FIXME: removing a record should adjust prev/next and mark those records as modified
|
|
|
|
// accordingly (also consider undo)
|
|
|
|
void CSMWorld::InfoCollection::removeRows (int index, int count)
|
|
|
|
{
|
|
|
|
Collection<Info, IdAccessor<Info> >::removeRows(index, count); // erase records only
|
|
|
|
|
2021-07-23 23:17:48 +00:00
|
|
|
for (std::unordered_map<std::string, std::vector<std::pair<std::string, int> > >::iterator iter
|
2021-07-23 06:05:58 +00:00
|
|
|
= mInfoIndex.begin(); iter != mInfoIndex.end();)
|
|
|
|
{
|
|
|
|
for (std::vector<std::pair<std::string, int> >::iterator it = iter->second.begin();
|
|
|
|
it != iter->second.end();)
|
|
|
|
{
|
|
|
|
if (it->second >= index)
|
|
|
|
{
|
|
|
|
if (it->second >= index+count)
|
|
|
|
{
|
|
|
|
it->second -= count;
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
else
|
2022-07-12 08:45:11 +00:00
|
|
|
it = iter->second.erase(it);
|
2021-07-23 06:05:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for an empty vector
|
|
|
|
if (iter->second.empty())
|
|
|
|
mInfoIndex.erase(iter++);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMWorld::InfoCollection::appendBlankRecord (const std::string& id, UniversalId::Type type)
|
|
|
|
{
|
2022-05-29 11:25:17 +00:00
|
|
|
auto record2 = std::make_unique<Record<Info>>();
|
2021-07-23 06:05:58 +00:00
|
|
|
|
|
|
|
record2->mState = Record<Info>::State_ModifiedOnly;
|
|
|
|
record2->mModified.blank();
|
|
|
|
|
|
|
|
record2->get().mId = id;
|
|
|
|
|
|
|
|
insertRecord(std::move(record2), getInsertIndex(id, type, nullptr), type); // call InfoCollection::insertRecord()
|
|
|
|
}
|
|
|
|
|
2021-09-04 16:07:23 +00:00
|
|
|
int CSMWorld::InfoCollection::searchId(std::string_view id) const
|
2021-07-23 06:05:58 +00:00
|
|
|
{
|
|
|
|
std::string::size_type separator = id.find_last_of('#');
|
|
|
|
|
|
|
|
if (separator == std::string::npos)
|
2021-09-04 16:07:23 +00:00
|
|
|
throw std::runtime_error("invalid info ID: " + std::string(id));
|
2021-07-23 06:05:58 +00:00
|
|
|
|
|
|
|
return getInfoIndex(id.substr(separator+1), id.substr(0, separator));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMWorld::InfoCollection::appendRecord (std::unique_ptr<RecordBase> record, UniversalId::Type type)
|
|
|
|
{
|
|
|
|
int index = getInsertIndex(static_cast<Record<Info>*>(record.get())->get().mId, type, record.get());
|
|
|
|
|
|
|
|
insertRecord(std::move(record), index, type);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSMWorld::InfoCollection::insertRecord (std::unique_ptr<RecordBase> record, int index,
|
|
|
|
UniversalId::Type type)
|
|
|
|
{
|
|
|
|
int size = static_cast<int>(getRecords().size());
|
|
|
|
|
|
|
|
std::string id = static_cast<Record<Info>*>(record.get())->get().mId;
|
|
|
|
std::string::size_type separator = id.find_last_of('#');
|
|
|
|
|
|
|
|
if (separator == std::string::npos)
|
|
|
|
throw std::runtime_error("invalid info ID: " + id);
|
|
|
|
|
|
|
|
Collection<Info, IdAccessor<Info> >::insertRecord(std::move(record), index, type); // add records only
|
|
|
|
|
|
|
|
// adjust index
|
|
|
|
if (index < size-1)
|
|
|
|
{
|
2021-07-23 23:17:48 +00:00
|
|
|
for (std::unordered_map<std::string, std::vector<std::pair<std::string, int> > >::iterator iter
|
2021-07-23 06:05:58 +00:00
|
|
|
= mInfoIndex.begin(); iter != mInfoIndex.end(); ++iter)
|
|
|
|
{
|
|
|
|
for (std::vector<std::pair<std::string, int> >::iterator it = iter->second.begin();
|
|
|
|
it != iter->second.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->second >= index)
|
|
|
|
++(it->second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// get iterator for existing topic or a new topic
|
|
|
|
std::string lowerId = Misc::StringUtils::lowerCase(id);
|
2021-07-23 23:17:48 +00:00
|
|
|
std::pair<std::unordered_map<std::string, std::vector<std::pair<std::string, int> > >::iterator, bool> res
|
2021-07-23 06:05:58 +00:00
|
|
|
= mInfoIndex.insert(
|
2021-07-23 23:17:48 +00:00
|
|
|
std::make_pair(lowerId.substr(0, separator),
|
2021-07-23 06:05:58 +00:00
|
|
|
std::vector<std::pair<std::string, int> >())); // empty vector
|
|
|
|
|
|
|
|
// insert info and index
|
|
|
|
res.first->second.push_back(std::make_pair(lowerId.substr(separator+1), index));
|
|
|
|
}
|