2010-08-03 14:14:04 +02:00
|
|
|
|
|
|
|
#include "weapon.hpp"
|
|
|
|
|
|
|
|
#include <components/esm/loadweap.hpp>
|
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
#include "../mwbase/world.hpp"
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2010-08-03 14:14:04 +02:00
|
|
|
#include "../mwworld/ptr.hpp"
|
2010-08-07 20:25:17 +02:00
|
|
|
#include "../mwworld/actiontake.hpp"
|
2012-05-16 16:08:55 +02:00
|
|
|
#include "../mwworld/actionequip.hpp"
|
2012-03-13 15:35:06 +01:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2012-06-29 16:48:50 +02:00
|
|
|
#include "../mwworld/cellstore.hpp"
|
2012-07-03 13:15:20 +02:00
|
|
|
#include "../mwworld/physicssystem.hpp"
|
2012-04-16 22:58:16 +02:00
|
|
|
|
|
|
|
#include "../mwgui/window_manager.hpp"
|
|
|
|
#include "../mwgui/tooltips.hpp"
|
2010-08-03 14:14:04 +02:00
|
|
|
|
2012-01-27 15:11:02 +01:00
|
|
|
#include "../mwrender/objects.hpp"
|
2012-07-03 13:15:20 +02:00
|
|
|
#include "../mwrender/renderinginterface.hpp"
|
2010-08-14 10:02:54 +02:00
|
|
|
|
2012-03-13 20:31:01 +02:00
|
|
|
#include "../mwsound/soundmanager.hpp"
|
|
|
|
|
2010-08-03 14:14:04 +02:00
|
|
|
namespace MWClass
|
|
|
|
{
|
2011-11-11 23:01:12 -05:00
|
|
|
void Weapon::insertObjectRendering (const MWWorld::Ptr& ptr, MWRender::RenderingInterface& renderingInterface) const
|
2010-08-14 10:02:54 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2010-08-14 10:02:54 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
assert (ref->base != NULL);
|
|
|
|
const std::string &model = ref->base->model;
|
2012-01-27 15:11:02 +01:00
|
|
|
|
2010-08-14 10:02:54 +02:00
|
|
|
if (!model.empty())
|
|
|
|
{
|
2011-11-19 01:01:19 -05:00
|
|
|
MWRender::Objects& objects = renderingInterface.getObjects();
|
2011-11-11 23:01:12 -05:00
|
|
|
objects.insertBegin(ptr, ptr.getRefData().isEnabled(), false);
|
|
|
|
objects.insertMesh(ptr, "meshes\\" + model);
|
2010-08-14 10:02:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
void Weapon::insertObject(const MWWorld::Ptr& ptr, MWWorld::PhysicsSystem& physics) const
|
2011-11-11 23:01:12 -05:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2011-11-11 23:01:12 -05:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
|
|
|
|
const std::string &model = ref->base->model;
|
|
|
|
assert (ref->base != NULL);
|
|
|
|
if(!model.empty()){
|
2011-11-17 19:38:52 -05:00
|
|
|
physics.insertObjectPhysics(ptr, "meshes\\" + model);
|
2011-11-11 23:01:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-08-03 17:11:41 +02:00
|
|
|
std::string Weapon::getName (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2010-08-03 17:11:41 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->name;
|
|
|
|
}
|
|
|
|
|
2010-08-07 20:25:17 +02:00
|
|
|
boost::shared_ptr<MWWorld::Action> Weapon::activate (const MWWorld::Ptr& ptr,
|
2012-04-23 15:27:03 +02:00
|
|
|
const MWWorld::Ptr& actor) const
|
2010-08-07 20:25:17 +02:00
|
|
|
{
|
2012-04-23 15:27:03 +02:00
|
|
|
MWBase::Environment::get().getSoundManager()->playSound3D (ptr, getUpSoundId(ptr), 1.0, 1.0, MWSound::Play_NoTrack);
|
2012-03-13 20:31:01 +02:00
|
|
|
|
2010-08-07 20:25:17 +02:00
|
|
|
return boost::shared_ptr<MWWorld::Action> (
|
|
|
|
new MWWorld::ActionTake (ptr));
|
|
|
|
}
|
|
|
|
|
2010-08-03 14:14:04 +02:00
|
|
|
bool Weapon::hasItemHealth (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Weapon::getItemMaxHealth (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2010-08-03 14:14:04 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->data.health;
|
|
|
|
}
|
|
|
|
|
2010-08-05 15:40:03 +02:00
|
|
|
std::string Weapon::getScript (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2010-08-05 15:40:03 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->script;
|
|
|
|
}
|
|
|
|
|
2012-03-13 15:35:06 +01:00
|
|
|
std::pair<std::vector<int>, bool> Weapon::getEquipmentSlots (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-03-13 15:35:06 +01:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
std::vector<int> slots;
|
|
|
|
bool stack = false;
|
|
|
|
|
|
|
|
if (ref->base->data.type==ESM::Weapon::Arrow || ref->base->data.type==ESM::Weapon::Bolt)
|
|
|
|
{
|
|
|
|
slots.push_back (int (MWWorld::InventoryStore::Slot_Ammunition));
|
|
|
|
stack = true;
|
|
|
|
}
|
|
|
|
else if (ref->base->data.type==ESM::Weapon::MarksmanThrown)
|
|
|
|
{
|
|
|
|
slots.push_back (int (MWWorld::InventoryStore::Slot_CarriedRight));
|
|
|
|
stack = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
slots.push_back (int (MWWorld::InventoryStore::Slot_CarriedRight));
|
|
|
|
|
|
|
|
return std::make_pair (slots, stack);
|
|
|
|
}
|
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
int Weapon::getEquipmentSkill (const MWWorld::Ptr& ptr) const
|
2012-03-13 15:35:06 +01:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-03-13 15:35:06 +01:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
const int size = 12;
|
|
|
|
|
|
|
|
static const int sMapping[size][2] =
|
|
|
|
{
|
|
|
|
{ ESM::Weapon::ShortBladeOneHand, ESM::Skill::ShortBlade },
|
|
|
|
{ ESM::Weapon::LongBladeOneHand, ESM::Skill::LongBlade },
|
|
|
|
{ ESM::Weapon::LongBladeTwoHand, ESM::Skill::LongBlade },
|
|
|
|
{ ESM::Weapon::BluntOneHand, ESM::Skill::BluntWeapon },
|
|
|
|
{ ESM::Weapon::BluntTwoClose, ESM::Skill::BluntWeapon },
|
|
|
|
{ ESM::Weapon::BluntTwoWide, ESM::Skill::BluntWeapon },
|
|
|
|
{ ESM::Weapon::SpearTwoWide, ESM::Skill::Spear },
|
|
|
|
{ ESM::Weapon::AxeOneHand, ESM::Skill::Axe },
|
|
|
|
{ ESM::Weapon::AxeTwoHand, ESM::Skill::Axe },
|
|
|
|
{ ESM::Weapon::MarksmanBow, ESM::Skill::Marksman },
|
|
|
|
{ ESM::Weapon::MarksmanCrossbow, ESM::Skill::Marksman },
|
|
|
|
{ ESM::Weapon::MarksmanThrown, ESM::Skill::Marksman }
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i=0; i<size; ++i)
|
|
|
|
if (sMapping[i][0]==ref->base->data.type)
|
|
|
|
return sMapping[i][1];
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-04-07 19:53:49 +02:00
|
|
|
int Weapon::getValue (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-04-07 19:53:49 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->data.value;
|
|
|
|
}
|
2012-03-13 15:35:06 +01:00
|
|
|
|
2010-08-03 14:14:04 +02:00
|
|
|
void Weapon::registerSelf()
|
|
|
|
{
|
|
|
|
boost::shared_ptr<Class> instance (new Weapon);
|
|
|
|
|
|
|
|
registerClass (typeid (ESM::Weapon).name(), instance);
|
|
|
|
}
|
2012-03-13 20:31:01 +02:00
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
std::string Weapon::getUpSoundId (const MWWorld::Ptr& ptr) const
|
2012-03-13 20:31:01 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-03-13 20:31:01 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
int type = ref->base->data.type;
|
|
|
|
// Ammo
|
|
|
|
if (type == 12 || type == 13)
|
|
|
|
{
|
|
|
|
return std::string("Item Ammo Up");
|
|
|
|
}
|
|
|
|
// Bow
|
|
|
|
if (type == 9)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Bow Up");
|
|
|
|
}
|
|
|
|
// Crossbow
|
|
|
|
if (type == 10)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Crossbow Up");
|
|
|
|
}
|
|
|
|
// Longblades, One hand and Two
|
|
|
|
if (type == 1 || type == 2)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Longblade Up");
|
|
|
|
}
|
|
|
|
// Shortblade and thrown weapons
|
|
|
|
// thrown weapons may not be entirely correct
|
|
|
|
if (type == 0 || type == 11)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Shortblade Up");
|
|
|
|
}
|
|
|
|
// Spear
|
|
|
|
if (type == 6)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Spear Up");
|
|
|
|
}
|
|
|
|
// Blunts and Axes
|
|
|
|
if (type == 3 || type == 4 || type == 5 || type == 7 || type == 8)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Blunt Up");
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::string("Item Misc Up");
|
|
|
|
}
|
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
std::string Weapon::getDownSoundId (const MWWorld::Ptr& ptr) const
|
2012-03-13 20:31:01 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-03-13 20:31:01 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
int type = ref->base->data.type;
|
|
|
|
// Ammo
|
|
|
|
if (type == 12 || type == 13)
|
|
|
|
{
|
|
|
|
return std::string("Item Ammo Down");
|
|
|
|
}
|
|
|
|
// Bow
|
|
|
|
if (type == 9)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Bow Down");
|
|
|
|
}
|
|
|
|
// Crossbow
|
|
|
|
if (type == 10)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Crossbow Down");
|
|
|
|
}
|
|
|
|
// Longblades, One hand and Two
|
|
|
|
if (type == 1 || type == 2)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Longblade Down");
|
|
|
|
}
|
|
|
|
// Shortblade and thrown weapons
|
|
|
|
// thrown weapons may not be entirely correct
|
|
|
|
if (type == 0 || type == 11)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Shortblade Down");
|
|
|
|
}
|
|
|
|
// Spear
|
|
|
|
if (type == 6)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Spear Down");
|
|
|
|
}
|
|
|
|
// Blunts and Axes
|
|
|
|
if (type == 3 || type == 4 || type == 5 || type == 7 || type == 8)
|
|
|
|
{
|
|
|
|
return std::string("Item Weapon Blunt Down");
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::string("Item Misc Down");
|
|
|
|
}
|
2012-04-15 17:52:39 +02:00
|
|
|
|
|
|
|
std::string Weapon::getInventoryIcon (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-04-15 17:52:39 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->icon;
|
|
|
|
}
|
2012-05-11 10:40:40 +02:00
|
|
|
|
2012-04-16 22:58:16 +02:00
|
|
|
bool Weapon::hasToolTip (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-04-16 22:58:16 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return (ref->base->name != "");
|
|
|
|
}
|
|
|
|
|
2012-04-24 02:02:03 +02:00
|
|
|
MWGui::ToolTipInfo Weapon::getToolTipInfo (const MWWorld::Ptr& ptr) const
|
2012-04-16 22:58:16 +02:00
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-04-16 22:58:16 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
MWGui::ToolTipInfo info;
|
2012-05-12 13:46:03 +02:00
|
|
|
info.caption = ref->base->name + MWGui::ToolTips::getCountString(ptr.getRefData().getCount());
|
2012-04-16 22:58:16 +02:00
|
|
|
info.icon = ref->base->icon;
|
|
|
|
|
2012-04-24 02:02:03 +02:00
|
|
|
const ESMS::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
|
2012-04-16 22:58:16 +02:00
|
|
|
std::string text;
|
|
|
|
|
2012-04-17 15:31:16 +02:00
|
|
|
// weapon type & damage. arrows / bolts don't have his info.
|
|
|
|
if (ref->base->data.type < 12)
|
|
|
|
{
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sType")->str + " ";
|
2012-04-17 15:31:16 +02:00
|
|
|
|
|
|
|
std::map <int, std::pair <std::string, std::string> > mapping;
|
|
|
|
mapping[ESM::Weapon::ShortBladeOneHand] = std::make_pair("sSkillShortblade", "sOneHanded");
|
|
|
|
mapping[ESM::Weapon::LongBladeOneHand] = std::make_pair("sSkillLongblade", "sOneHanded");
|
|
|
|
mapping[ESM::Weapon::LongBladeTwoHand] = std::make_pair("sSkillLongblade", "sTwoHanded");
|
|
|
|
mapping[ESM::Weapon::BluntOneHand] = std::make_pair("sSkillBluntweapon", "sOneHanded");
|
|
|
|
mapping[ESM::Weapon::BluntTwoClose] = std::make_pair("sSkillBluntweapon", "sTwoHanded");
|
|
|
|
mapping[ESM::Weapon::BluntTwoWide] = std::make_pair("sSkillBluntweapon", "sTwoHanded");
|
|
|
|
mapping[ESM::Weapon::SpearTwoWide] = std::make_pair("sSkillSpear", "sTwoHanded");
|
|
|
|
mapping[ESM::Weapon::AxeOneHand] = std::make_pair("sSkillAxe", "sOneHanded");
|
|
|
|
mapping[ESM::Weapon::AxeTwoHand] = std::make_pair("sSkillAxe", "sTwoHanded");
|
|
|
|
mapping[ESM::Weapon::MarksmanBow] = std::make_pair("sSkillMarksman", "");
|
|
|
|
mapping[ESM::Weapon::MarksmanCrossbow] = std::make_pair("sSkillMarksman", "");
|
|
|
|
mapping[ESM::Weapon::MarksmanThrown] = std::make_pair("sSkillMarksman", "");
|
|
|
|
|
|
|
|
std::string type = mapping[ref->base->data.type].first;
|
|
|
|
std::string oneOrTwoHanded = mapping[ref->base->data.type].second;
|
|
|
|
|
2012-04-24 02:02:03 +02:00
|
|
|
text += store.gameSettings.search(type)->str +
|
|
|
|
((oneOrTwoHanded != "") ? ", " + store.gameSettings.search(oneOrTwoHanded)->str : "");
|
2012-04-17 15:31:16 +02:00
|
|
|
|
|
|
|
// weapon damage
|
|
|
|
if (ref->base->data.type >= 9)
|
|
|
|
{
|
|
|
|
// marksman
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sAttack")->str + ": "
|
2012-04-17 15:31:16 +02:00
|
|
|
+ MWGui::ToolTips::toString(static_cast<int>(ref->base->data.chop[0]))
|
|
|
|
+ " - " + MWGui::ToolTips::toString(static_cast<int>(ref->base->data.chop[1]));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Chop
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sChop")->str + ": "
|
2012-04-17 15:31:16 +02:00
|
|
|
+ MWGui::ToolTips::toString(static_cast<int>(ref->base->data.chop[0]))
|
|
|
|
+ " - " + MWGui::ToolTips::toString(static_cast<int>(ref->base->data.chop[1]));
|
|
|
|
// Slash
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sSlash")->str + ": "
|
2012-04-17 15:31:16 +02:00
|
|
|
+ MWGui::ToolTips::toString(static_cast<int>(ref->base->data.slash[0]))
|
|
|
|
+ " - " + MWGui::ToolTips::toString(static_cast<int>(ref->base->data.slash[1]));
|
|
|
|
// Thrust
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sThrust")->str + ": "
|
2012-04-17 15:31:16 +02:00
|
|
|
+ MWGui::ToolTips::toString(static_cast<int>(ref->base->data.thrust[0]))
|
|
|
|
+ " - " + MWGui::ToolTips::toString(static_cast<int>(ref->base->data.thrust[1]));
|
|
|
|
}
|
|
|
|
}
|
2012-04-16 22:58:16 +02:00
|
|
|
|
|
|
|
/// \todo store the current weapon health somewhere
|
2012-04-17 15:31:16 +02:00
|
|
|
if (ref->base->data.type < 11) // thrown weapons and arrows/bolts don't have health, only quantity
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sCondition")->str + ": " + MWGui::ToolTips::toString(ref->base->data.health);
|
2012-04-16 22:58:16 +02:00
|
|
|
|
2012-04-24 02:02:03 +02:00
|
|
|
text += "\n" + store.gameSettings.search("sWeight")->str + ": " + MWGui::ToolTips::toString(ref->base->data.weight);
|
|
|
|
text += MWGui::ToolTips::getValueString(ref->base->data.value, store.gameSettings.search("sValue")->str);
|
2012-04-16 22:58:16 +02:00
|
|
|
|
2012-04-30 00:57:41 +02:00
|
|
|
info.enchant = ref->base->enchant;
|
2012-04-18 16:53:56 +02:00
|
|
|
|
2012-04-24 02:02:03 +02:00
|
|
|
if (MWBase::Environment::get().getWindowManager()->getFullHelp()) {
|
2012-04-16 22:58:16 +02:00
|
|
|
text += MWGui::ToolTips::getMiscString(ref->ref.owner, "Owner");
|
|
|
|
text += MWGui::ToolTips::getMiscString(ref->base->script, "Script");
|
|
|
|
}
|
|
|
|
|
|
|
|
info.text = text;
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
2012-05-12 16:17:03 +02:00
|
|
|
|
|
|
|
std::string Weapon::getEnchantment (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
2012-06-29 18:54:23 +02:00
|
|
|
MWWorld::LiveCellRef<ESM::Weapon> *ref =
|
2012-05-12 16:17:03 +02:00
|
|
|
ptr.get<ESM::Weapon>();
|
|
|
|
|
|
|
|
return ref->base->enchant;
|
|
|
|
}
|
2012-05-16 16:08:55 +02:00
|
|
|
|
|
|
|
boost::shared_ptr<MWWorld::Action> Weapon::use (const MWWorld::Ptr& ptr) const
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getSoundManager()->playSound (getUpSoundId(ptr), 1.0, 1.0);
|
|
|
|
|
|
|
|
return boost::shared_ptr<MWWorld::Action>(new MWWorld::ActionEquip(ptr));
|
|
|
|
}
|
2010-08-03 14:14:04 +02:00
|
|
|
}
|