2012-11-22 12:30:02 +00:00
|
|
|
|
|
|
|
#include "viewmanager.hpp"
|
|
|
|
|
2012-11-22 14:09:04 +00:00
|
|
|
#include <map>
|
|
|
|
|
2013-05-08 01:33:42 +00:00
|
|
|
#include <QApplication>
|
|
|
|
#include <QDesktopWidget>
|
|
|
|
|
2012-11-22 13:10:23 +00:00
|
|
|
#include "../../model/doc/documentmanager.hpp"
|
2012-11-22 22:42:17 +00:00
|
|
|
#include "../../model/doc/document.hpp"
|
2012-11-22 13:10:23 +00:00
|
|
|
|
2013-02-10 16:21:25 +00:00
|
|
|
#include "../world/util.hpp"
|
2013-02-17 16:27:25 +00:00
|
|
|
#include "../world/enumdelegate.hpp"
|
|
|
|
#include "../world/vartypedelegate.hpp"
|
2013-05-29 11:38:35 +00:00
|
|
|
#include "../world/recordstatusdelegate.hpp"
|
2013-06-15 11:40:18 +00:00
|
|
|
#include "../settings/usersettingsdialog.hpp"
|
2013-02-10 16:21:25 +00:00
|
|
|
|
2012-11-22 12:30:02 +00:00
|
|
|
#include "view.hpp"
|
|
|
|
|
2013-03-02 13:57:41 +00:00
|
|
|
#include <QMessageBox>
|
2013-03-06 12:41:33 +00:00
|
|
|
#include <QPushButton>
|
2013-03-11 11:38:27 +00:00
|
|
|
#include <QtGui/QApplication>
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2012-11-22 14:09:04 +00:00
|
|
|
void CSVDoc::ViewManager::updateIndices()
|
|
|
|
{
|
|
|
|
std::map<CSMDoc::Document *, std::pair<int, int> > documents;
|
|
|
|
|
|
|
|
for (std::vector<View *>::const_iterator iter (mViews.begin()); iter!=mViews.end(); ++iter)
|
|
|
|
{
|
|
|
|
std::map<CSMDoc::Document *, std::pair<int, int> >::iterator document = documents.find ((*iter)->getDocument());
|
|
|
|
|
|
|
|
if (document==documents.end())
|
|
|
|
document =
|
|
|
|
documents.insert (
|
|
|
|
std::make_pair ((*iter)->getDocument(), std::make_pair (0, countViews ((*iter)->getDocument())))).
|
|
|
|
first;
|
|
|
|
|
|
|
|
(*iter)->setIndex (document->second.first++, document->second.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-22 13:10:23 +00:00
|
|
|
CSVDoc::ViewManager::ViewManager (CSMDoc::DocumentManager& documentManager)
|
2013-03-12 11:28:13 +00:00
|
|
|
: mDocumentManager (documentManager), mExitOnSaveStateChange(false), mUserWarned(false)
|
2012-11-22 12:30:02 +00:00
|
|
|
{
|
2013-03-23 12:13:53 +00:00
|
|
|
static const char *sSpecialisations[] =
|
|
|
|
{
|
|
|
|
"Combat", "Magic", "Stealth", 0
|
|
|
|
};
|
|
|
|
|
2013-03-24 14:50:29 +00:00
|
|
|
static const char *sAttributes[] =
|
|
|
|
{
|
|
|
|
"Strength", "Intelligence", "Willpower", "Agility", "Speed", "Endurance", "Personality",
|
|
|
|
"Luck", 0
|
|
|
|
};
|
|
|
|
|
2013-04-09 09:53:47 +00:00
|
|
|
static const char *sSpellTypes[] =
|
|
|
|
{
|
|
|
|
"Spell", "Ability", "Blight", "Disease", "Curse", "Power", 0
|
|
|
|
};
|
|
|
|
|
2013-05-13 10:51:27 +00:00
|
|
|
static const char *sApparatusTypes[] =
|
|
|
|
{
|
|
|
|
"Mortar & Pestle", "Albemic", "Calcinator", "Retort", 0
|
|
|
|
};
|
|
|
|
|
2013-05-14 12:21:30 +00:00
|
|
|
static const char *sArmorTypes[] =
|
|
|
|
{
|
|
|
|
"Helmet", "Cuirass", "Left Pauldron", "Right Pauldron", "Greaves", "Boots", "Left Gauntlet",
|
|
|
|
"Right Gauntlet", "Shield", "Left Bracer", "Right Bracer", 0
|
|
|
|
};
|
|
|
|
|
2013-05-15 09:37:46 +00:00
|
|
|
static const char *sClothingTypes[] =
|
|
|
|
{
|
|
|
|
"Pants", "Shoes", "Shirt", "Belt", "Robe", "Right Glove", "Left Glove", "Skirt", "Ring",
|
|
|
|
"Amulet", 0
|
|
|
|
};
|
|
|
|
|
2013-05-16 14:27:33 +00:00
|
|
|
static const char *sCreatureTypes[] =
|
|
|
|
{
|
|
|
|
"Creature", "Deadra", "Undead", "Humanoid", 0
|
|
|
|
};
|
|
|
|
|
2013-05-19 12:44:41 +00:00
|
|
|
static const char *sWeaponTypes[] =
|
|
|
|
{
|
|
|
|
"Short Blade 1H", "Long Blade 1H", "Long Blade 2H", "Blunt 1H", "Blunt 2H Close",
|
|
|
|
"Blunt 2H Wide", "Spear 2H", "Axe 1H", "Axe 2H", "Bow", "Crossbow", "Thrown", "Arrow",
|
|
|
|
"Bolt", 0
|
|
|
|
};
|
|
|
|
|
2013-02-10 16:21:25 +00:00
|
|
|
mDelegateFactories = new CSVWorld::CommandDelegateFactoryCollection;
|
2013-02-17 16:27:25 +00:00
|
|
|
|
2013-03-05 10:37:13 +00:00
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_GmstVarType,
|
2013-02-17 16:27:25 +00:00
|
|
|
new CSVWorld::VarTypeDelegateFactory (ESM::VT_None, ESM::VT_String, ESM::VT_Int, ESM::VT_Float));
|
2013-03-05 10:37:13 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_GlobalVarType,
|
|
|
|
new CSVWorld::VarTypeDelegateFactory (ESM::VT_Short, ESM::VT_Long, ESM::VT_Float));
|
2013-03-23 12:13:53 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_Specialisation,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sSpecialisations));
|
2013-03-24 14:50:29 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_Attribute,
|
2013-04-02 12:15:22 +00:00
|
|
|
new CSVWorld::EnumDelegateFactory (sAttributes, true));
|
2013-04-09 09:53:47 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_SpellType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sSpellTypes));
|
2013-05-13 10:51:27 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_ApparatusType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sApparatusTypes));
|
2013-05-14 12:21:30 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_ArmorType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sArmorTypes));
|
2013-05-15 09:37:46 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_ClothingType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sClothingTypes));
|
2013-05-16 14:27:33 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_CreatureType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sCreatureTypes));
|
2013-05-19 12:44:41 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_WeaponType,
|
|
|
|
new CSVWorld::EnumDelegateFactory (sWeaponTypes));
|
2013-05-29 11:38:35 +00:00
|
|
|
|
|
|
|
mDelegateFactories->add (CSMWorld::ColumnBase::Display_RecordState,
|
|
|
|
new CSVWorld::RecordStatusDelegateFactory() );
|
2013-06-15 11:40:18 +00:00
|
|
|
|
|
|
|
connect (&CSMSettings::UserSettings::instance(), SIGNAL (signalUpdateEditorSetting (const QString &, const QString &)),
|
|
|
|
this, SLOT (slotUpdateEditorSetting (const QString &, const QString &)));
|
2012-11-22 12:30:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CSVDoc::ViewManager::~ViewManager()
|
|
|
|
{
|
2013-02-10 16:21:25 +00:00
|
|
|
delete mDelegateFactories;
|
|
|
|
|
2012-11-22 12:30:02 +00:00
|
|
|
for (std::vector<View *>::iterator iter (mViews.begin()); iter!=mViews.end(); ++iter)
|
|
|
|
delete *iter;
|
|
|
|
}
|
|
|
|
|
|
|
|
CSVDoc::View *CSVDoc::ViewManager::addView (CSMDoc::Document *document)
|
|
|
|
{
|
2012-11-22 22:42:17 +00:00
|
|
|
if (countViews (document)==0)
|
|
|
|
{
|
|
|
|
// new document
|
|
|
|
connect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)),
|
|
|
|
this, SLOT (documentStateChanged (int, CSMDoc::Document *)));
|
2012-11-23 09:25:34 +00:00
|
|
|
|
2012-11-23 11:20:35 +00:00
|
|
|
connect (document, SIGNAL (progress (int, int, int, int, CSMDoc::Document *)),
|
|
|
|
this, SLOT (progress (int, int, int, int, CSMDoc::Document *)));
|
2012-11-22 22:42:17 +00:00
|
|
|
}
|
|
|
|
|
2013-03-03 03:34:55 +00:00
|
|
|
View *view = new View (*this, document, countViews (document)+1);
|
2012-11-22 12:30:02 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
|
2012-11-22 12:30:02 +00:00
|
|
|
mViews.push_back (view);
|
|
|
|
|
|
|
|
view->show();
|
|
|
|
|
2012-11-23 13:05:49 +00:00
|
|
|
connect (view, SIGNAL (newDocumentRequest ()), this, SIGNAL (newDocumentRequest()));
|
2013-02-02 15:14:58 +00:00
|
|
|
connect (view, SIGNAL (loadDocumentRequest ()), this, SIGNAL (loadDocumentRequest()));
|
2012-11-23 13:05:49 +00:00
|
|
|
|
2012-11-22 14:09:04 +00:00
|
|
|
updateIndices();
|
|
|
|
|
2012-11-22 12:30:02 +00:00
|
|
|
return view;
|
2012-11-22 13:10:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSVDoc::ViewManager::countViews (const CSMDoc::Document *document) const
|
|
|
|
{
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (std::vector<View *>::const_iterator iter (mViews.begin()); iter!=mViews.end(); ++iter)
|
|
|
|
if ((*iter)->getDocument()==document)
|
|
|
|
++count;
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVDoc::ViewManager::closeRequest (View *view)
|
|
|
|
{
|
|
|
|
std::vector<View *>::iterator iter = std::find (mViews.begin(), mViews.end(), view);
|
|
|
|
|
2013-03-02 13:57:41 +00:00
|
|
|
bool continueWithClose = true;
|
|
|
|
|
2012-11-22 13:10:23 +00:00
|
|
|
if (iter!=mViews.end())
|
|
|
|
{
|
|
|
|
bool last = countViews (view->getDocument())<=1;
|
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
if (last)
|
|
|
|
continueWithClose = notifySaveOnClose (view);
|
|
|
|
else
|
2013-03-02 13:57:41 +00:00
|
|
|
{
|
2013-03-06 12:41:33 +00:00
|
|
|
(*iter)->deleteLater();
|
|
|
|
mViews.erase (iter);
|
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
updateIndices();
|
2013-03-02 13:57:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return continueWithClose;
|
|
|
|
}
|
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
bool CSVDoc::ViewManager::notifySaveOnClose (CSVDoc::View *view)
|
|
|
|
{
|
|
|
|
bool result = true;
|
|
|
|
CSMDoc::Document *document = view->getDocument();
|
|
|
|
|
|
|
|
//notify user of saving in progress
|
|
|
|
if ( (document->getState() & CSMDoc::State_Saving) )
|
|
|
|
result = showSaveInProgressMessageBox (view);
|
|
|
|
|
|
|
|
//notify user of unsaved changes and process response
|
|
|
|
else if ( document->getState() & CSMDoc::State_Modified)
|
|
|
|
result = showModifiedDocumentMessageBox (view);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVDoc::ViewManager::showModifiedDocumentMessageBox (CSVDoc::View *view)
|
2013-03-02 13:57:41 +00:00
|
|
|
{
|
|
|
|
QMessageBox messageBox;
|
2013-03-12 11:28:13 +00:00
|
|
|
CSMDoc::Document *document = view->getDocument();
|
2013-03-02 13:57:41 +00:00
|
|
|
|
|
|
|
messageBox.setText ("The document has been modified.");
|
|
|
|
messageBox.setInformativeText ("Do you want to save your changes?");
|
|
|
|
messageBox.setStandardButtons (QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel);
|
|
|
|
messageBox.setDefaultButton (QMessageBox::Save);
|
|
|
|
|
|
|
|
bool retVal = true;
|
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
connect (this, SIGNAL (closeMessageBox()), &messageBox, SLOT (close()));
|
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
connect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
2013-03-06 12:41:33 +00:00
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
mUserWarned = true;
|
|
|
|
int response = messageBox.exec();
|
2013-03-06 12:41:33 +00:00
|
|
|
mUserWarned = false;
|
|
|
|
|
|
|
|
switch (response)
|
2013-03-02 13:57:41 +00:00
|
|
|
{
|
|
|
|
case QMessageBox::Save:
|
2013-03-02 18:49:26 +00:00
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
document->save();
|
|
|
|
mExitOnSaveStateChange = true;
|
2013-03-06 12:41:33 +00:00
|
|
|
retVal = false;
|
2013-03-02 13:57:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case QMessageBox::Discard:
|
2013-03-06 12:41:33 +00:00
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
disconnect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
2013-03-02 13:57:41 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case QMessageBox::Cancel:
|
2013-03-06 12:41:33 +00:00
|
|
|
|
|
|
|
//disconnect to prevent unintended view closures
|
2013-03-12 11:28:13 +00:00
|
|
|
disconnect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
2013-03-02 13:57:41 +00:00
|
|
|
retVal = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return retVal;
|
|
|
|
}
|
2012-11-22 13:10:23 +00:00
|
|
|
|
2013-03-12 11:28:13 +00:00
|
|
|
bool CSVDoc::ViewManager::showSaveInProgressMessageBox (CSVDoc::View *view)
|
2013-03-02 13:57:41 +00:00
|
|
|
{
|
|
|
|
QMessageBox messageBox;
|
2013-03-12 11:28:13 +00:00
|
|
|
CSMDoc::Document *document = view->getDocument();
|
2013-03-02 13:57:41 +00:00
|
|
|
|
|
|
|
messageBox.setText ("The document is currently being saved.");
|
2013-03-06 12:41:33 +00:00
|
|
|
messageBox.setInformativeText("Do you want to close now and abort saving, or wait until saving has completed?");
|
|
|
|
|
|
|
|
QPushButton* waitButton = messageBox.addButton (tr("Wait"), QMessageBox::YesRole);
|
|
|
|
QPushButton* closeButton = messageBox.addButton (tr("Close Now"), QMessageBox::RejectRole);
|
|
|
|
QPushButton* cancelButton = messageBox.addButton (tr("Cancel"), QMessageBox::NoRole);
|
|
|
|
|
|
|
|
messageBox.setDefaultButton (waitButton);
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2013-03-02 15:22:44 +00:00
|
|
|
bool retVal = true;
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
//Connections shut down message box if operation ends before user makes a decision.
|
2013-03-12 11:28:13 +00:00
|
|
|
connect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
2013-03-06 12:41:33 +00:00
|
|
|
connect (this, SIGNAL (closeMessageBox()), &messageBox, SLOT (close()));
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
//set / clear the user warned flag to indicate whether or not the message box is currently active.
|
|
|
|
mUserWarned = true;
|
|
|
|
messageBox.exec();
|
|
|
|
mUserWarned = false;
|
2013-03-02 18:49:26 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
//if closed by the warning handler, defaults to the RejectRole button (closeButton)
|
|
|
|
if (messageBox.clickedButton() == waitButton)
|
|
|
|
{
|
|
|
|
//save the View iterator for shutdown after the save operation ends
|
2013-03-12 11:28:13 +00:00
|
|
|
mExitOnSaveStateChange = true;
|
2013-03-06 12:41:33 +00:00
|
|
|
retVal = false;
|
|
|
|
}
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
else if (messageBox.clickedButton() == closeButton)
|
|
|
|
{
|
|
|
|
//disconnect to avoid segmentation fault
|
2013-03-12 11:28:13 +00:00
|
|
|
disconnect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
|
|
|
|
|
|
|
view->abortOperation(CSMDoc::State_Saving);
|
|
|
|
mExitOnSaveStateChange = true;
|
2013-03-06 12:41:33 +00:00
|
|
|
}
|
2013-03-02 13:57:41 +00:00
|
|
|
|
2013-03-06 12:41:33 +00:00
|
|
|
else if (messageBox.clickedButton() == cancelButton)
|
|
|
|
{
|
|
|
|
//abort shutdown, allow save to complete
|
|
|
|
//disconnection to prevent unintended view closures
|
2013-03-12 11:28:13 +00:00
|
|
|
mExitOnSaveStateChange = false;
|
|
|
|
disconnect (document, SIGNAL (stateChanged (int, CSMDoc::Document *)), this, SLOT (onExitWarningHandler(int, CSMDoc::Document *)));
|
2013-03-06 12:41:33 +00:00
|
|
|
retVal = false;
|
2012-11-22 13:10:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-02 13:57:41 +00:00
|
|
|
return retVal;
|
2012-11-22 22:42:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVDoc::ViewManager::documentStateChanged (int state, CSMDoc::Document *document)
|
|
|
|
{
|
|
|
|
for (std::vector<View *>::const_iterator iter (mViews.begin()); iter!=mViews.end(); ++iter)
|
|
|
|
if ((*iter)->getDocument()==document)
|
|
|
|
(*iter)->updateDocumentState();
|
2012-11-22 23:36:01 +00:00
|
|
|
}
|
2012-11-23 09:25:34 +00:00
|
|
|
|
2012-11-23 11:20:35 +00:00
|
|
|
void CSVDoc::ViewManager::progress (int current, int max, int type, int threads, CSMDoc::Document *document)
|
2012-11-23 09:25:34 +00:00
|
|
|
{
|
|
|
|
for (std::vector<View *>::const_iterator iter (mViews.begin()); iter!=mViews.end(); ++iter)
|
|
|
|
if ((*iter)->getDocument()==document)
|
2012-11-23 11:20:35 +00:00
|
|
|
(*iter)->updateProgress (current, max, type, threads);
|
2013-02-23 03:53:32 +00:00
|
|
|
}
|
2013-03-06 12:41:33 +00:00
|
|
|
|
2013-03-11 11:38:27 +00:00
|
|
|
void CSVDoc::ViewManager::onExitWarningHandler (int state, CSMDoc::Document *document)
|
2013-03-06 12:41:33 +00:00
|
|
|
{
|
|
|
|
if ( !(state & CSMDoc::State_Saving) )
|
|
|
|
{
|
|
|
|
//if the user is being warned (message box is active), shut down the message box,
|
|
|
|
//as there is no save operation currently running
|
|
|
|
if ( mUserWarned )
|
|
|
|
emit closeMessageBox();
|
|
|
|
|
|
|
|
//otherwise, the user has closed the message box before the save operation ended.
|
2013-03-11 11:38:27 +00:00
|
|
|
//exit the application
|
2013-03-12 11:28:13 +00:00
|
|
|
else if (mExitOnSaveStateChange)
|
|
|
|
QApplication::instance()->exit();
|
2013-03-06 12:41:33 +00:00
|
|
|
}
|
|
|
|
}
|
2013-03-12 11:28:13 +00:00
|
|
|
|
|
|
|
void CSVDoc::ViewManager::exitApplication (CSVDoc::View *view)
|
|
|
|
{
|
|
|
|
if (notifySaveOnClose (view))
|
|
|
|
QApplication::instance()->exit();
|
|
|
|
}
|
2013-06-15 11:40:18 +00:00
|
|
|
|
|
|
|
void CSVDoc::ViewManager::slotUpdateEditorSetting (const QString &settingName, const QString &settingValue)
|
|
|
|
{
|
2013-06-20 23:06:25 +00:00
|
|
|
if (settingName == "Record Status Display" ||
|
|
|
|
settingName == "Width" || settingName == "Height")
|
|
|
|
{
|
2013-06-15 11:40:18 +00:00
|
|
|
foreach (CSVDoc::View *view, mViews)
|
|
|
|
view->updateEditorSetting (settingName, settingValue);
|
2013-06-20 23:06:25 +00:00
|
|
|
}
|
2013-06-15 11:40:18 +00:00
|
|
|
}
|