2017-08-21 05:46:59 +00:00
|
|
|
#include "CAuxWeapon.hpp"
|
2017-09-10 00:36:21 +00:00
|
|
|
#include "GameGlobalObjects.hpp"
|
|
|
|
#include "CSimplePool.hpp"
|
|
|
|
#include "CWaveBuster.hpp"
|
|
|
|
#include "CNewFlameThrower.hpp"
|
|
|
|
#include "CEnergyProjectile.hpp"
|
2017-08-21 05:46:59 +00:00
|
|
|
|
|
|
|
namespace urde
|
|
|
|
{
|
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
static const CCameraShakeData skHardShake = { 0.3f, 100.f, 0, zeus::CVector3f::skZero,
|
|
|
|
{}, {1, {0, 0.f, 0.f, 0.3f, -2.f}, {1, 0.f, 0.f, 0.05f, 0.5f}}, {} };
|
2017-08-21 05:46:59 +00:00
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
CAuxWeapon::CAuxWeapon(TUniqueId playerId)
|
|
|
|
: x0_missile(g_SimplePool->GetObj("Missile")),
|
|
|
|
xc_flameMuzzle(g_SimplePool->GetObj("FlameMuzzle")),
|
|
|
|
x18_busterMuzzle(g_SimplePool->GetObj("BusterMuzzle")),
|
|
|
|
x6c_playerId(playerId)
|
|
|
|
{
|
|
|
|
x0_missile.GetObj();
|
|
|
|
xc_flameMuzzle.GetObj();
|
|
|
|
x18_busterMuzzle.GetObj();
|
|
|
|
x80_24_isLoaded = false;
|
|
|
|
InitComboData();
|
2017-08-21 05:46:59 +00:00
|
|
|
}
|
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
static const char* skComboNames[] =
|
2017-08-25 06:18:09 +00:00
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
"SuperMissile",
|
|
|
|
"IceCombo",
|
|
|
|
"WaveBuster",
|
|
|
|
"FlameThrower",
|
|
|
|
"SuperMissile"
|
|
|
|
};
|
|
|
|
|
|
|
|
void CAuxWeapon::InitComboData()
|
|
|
|
{
|
|
|
|
for (int i=0 ; i<5 ; ++i)
|
|
|
|
x28_combos.push_back(g_SimplePool->GetObj(skComboNames[i]));
|
|
|
|
}
|
2017-08-25 06:18:09 +00:00
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
void CAuxWeapon::AcceptScriptMsg(EScriptObjectMessage msg, TUniqueId sender, CStateManager& mgr)
|
|
|
|
{
|
|
|
|
if (msg == EScriptObjectMessage::Deleted)
|
|
|
|
{
|
|
|
|
DeleteFlameThrower(mgr);
|
|
|
|
DeleteWaveBusterBeam(mgr);
|
|
|
|
}
|
2017-08-25 06:18:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CAuxWeapon::IsComboFxActive(const CStateManager& mgr) const
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
switch (x74_firingBeamId)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
if (const CEntity* ent = mgr.GetObjectById(x70_waveBusterId))
|
|
|
|
return static_cast<const CWaveBuster*>(ent)->IsFiring();
|
|
|
|
break;
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
if (const CEntity* ent = mgr.GetObjectById(x6e_flameThrowerId))
|
|
|
|
return static_cast<const CNewFlameThrower*>(ent)->IsFiring();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2017-08-25 06:18:09 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-08-31 02:42:37 +00:00
|
|
|
void CAuxWeapon::Load(CPlayerState::EBeamId curBeam, CStateManager& mgr)
|
2017-08-25 06:18:09 +00:00
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
x80_24_isLoaded = false;
|
|
|
|
switch (x78_loadBeamId)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
DeleteWaveBusterBeam(mgr);
|
|
|
|
break;
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
DeleteFlameThrower(mgr);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
x28_combos[int(x78_loadBeamId)].Unlock();
|
|
|
|
x28_combos[int(curBeam)].Lock();
|
|
|
|
x78_loadBeamId = curBeam;
|
|
|
|
LoadIdle();
|
2017-08-25 06:18:09 +00:00
|
|
|
}
|
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
void CAuxWeapon::StopComboFx(CStateManager& mgr, bool deactivate)
|
2017-08-26 04:36:25 +00:00
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
switch (x74_firingBeamId)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
{
|
|
|
|
auto* wb = static_cast<CWaveBuster*>(mgr.ObjectById(x70_waveBusterId));
|
|
|
|
if (wb)
|
|
|
|
{
|
|
|
|
wb->ResetBeam(deactivate);
|
|
|
|
DeleteWaveBusterBeam(mgr);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
{
|
|
|
|
auto* ft = static_cast<CNewFlameThrower*>(mgr.ObjectById(x6e_flameThrowerId));
|
|
|
|
if (ft)
|
|
|
|
{
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(false);
|
|
|
|
if (ft->IsFiring())
|
|
|
|
{
|
|
|
|
ft->Reset(mgr, deactivate);
|
|
|
|
FreeComboVoiceId();
|
|
|
|
}
|
|
|
|
else if (ft->GetActive() && deactivate)
|
|
|
|
{
|
|
|
|
ft->Reset(mgr, deactivate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2017-08-26 04:36:25 +00:00
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
if (deactivate)
|
|
|
|
{
|
|
|
|
x74_firingBeamId = CPlayerState::EBeamId::Invalid;
|
|
|
|
x68_ammoConsumeTimer = 0.f;
|
|
|
|
}
|
2017-08-26 04:36:25 +00:00
|
|
|
}
|
|
|
|
|
2017-09-02 04:06:05 +00:00
|
|
|
bool CAuxWeapon::UpdateComboFx(float dt, const zeus::CVector3f& scale, const zeus::CVector3f& pos,
|
|
|
|
const zeus::CTransform& xf, CStateManager& mgr)
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
if (!x80_24_isLoaded || x74_firingBeamId == CPlayerState::EBeamId::Invalid)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool firing = false;
|
|
|
|
if (x7c_comboSfx && !CSfxManager::IsPlaying(x7c_comboSfx))
|
|
|
|
FreeComboVoiceId();
|
|
|
|
|
|
|
|
switch (x74_firingBeamId)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
{
|
|
|
|
bool firingFx = false;
|
|
|
|
if (x74_firingBeamId == CPlayerState::EBeamId::Wave)
|
|
|
|
{
|
|
|
|
auto* wb = static_cast<CWaveBuster*>(mgr.ObjectById(x70_waveBusterId));
|
|
|
|
if (wb && wb->IsFiring())
|
|
|
|
{
|
|
|
|
wb->UpdateFx(xf, dt, mgr);
|
|
|
|
firing = true;
|
|
|
|
firingFx = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DeleteWaveBusterBeam(mgr);
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto* ft = static_cast<CNewFlameThrower*>(mgr.ObjectById(x6e_flameThrowerId));
|
|
|
|
bool needsDelete = true;
|
|
|
|
if (ft)
|
|
|
|
{
|
|
|
|
firingFx = ft->CanRenderAuxEffects();
|
|
|
|
if (ft->GetActive())
|
|
|
|
{
|
|
|
|
ft->UpdateFx(xf, dt, mgr);
|
|
|
|
firing = ft->IsFiring();
|
|
|
|
}
|
|
|
|
if (x6e_flameThrowerId != kInvalidUniqueId)
|
|
|
|
needsDelete = ft->AreEffectsFinished();
|
|
|
|
}
|
|
|
|
if (needsDelete)
|
|
|
|
{
|
|
|
|
DeleteFlameThrower(mgr);
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (firingFx)
|
|
|
|
{
|
|
|
|
x68_ammoConsumeTimer += dt;
|
|
|
|
if (mgr.GetPlayerState()->GetItemAmount(CPlayerState::EItemType::Missiles) > 0)
|
|
|
|
{
|
|
|
|
if (x68_ammoConsumeTimer >= mgr.GetPlayerState()->GetComboFireAmmoPeriod())
|
|
|
|
{
|
|
|
|
mgr.GetPlayerState()->DecrPickup(CPlayerState::EItemType::Missiles, 1);
|
|
|
|
x68_ammoConsumeTimer = 0.f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mgr.GetPlayerState()->GetItemAmount(CPlayerState::EItemType::Missiles) == 0)
|
|
|
|
StopComboFx(mgr, false);
|
|
|
|
|
|
|
|
x24_muzzleFxGen->SetGlobalTranslation(pos);
|
|
|
|
x24_muzzleFxGen->SetGlobalScale(scale);
|
|
|
|
x24_muzzleFxGen->SetParticleEmission(firingFx);
|
|
|
|
x24_muzzleFxGen->Update(dt);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return firing;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CAuxWeapon::FreeComboVoiceId()
|
|
|
|
{
|
|
|
|
CSfxManager::SfxStop(x7c_comboSfx);
|
|
|
|
x7c_comboSfx.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CAuxWeapon::DeleteFlameThrower(CStateManager& mgr)
|
|
|
|
{
|
|
|
|
FreeComboVoiceId();
|
|
|
|
if (x6e_flameThrowerId != kInvalidUniqueId)
|
|
|
|
{
|
|
|
|
mgr.FreeScriptObject(x6e_flameThrowerId);
|
|
|
|
x6e_flameThrowerId = kInvalidUniqueId;
|
|
|
|
x74_firingBeamId = CPlayerState::EBeamId::Invalid;
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(false);
|
|
|
|
}
|
2017-09-02 04:06:05 +00:00
|
|
|
}
|
|
|
|
|
2017-09-10 00:36:21 +00:00
|
|
|
void CAuxWeapon::CreateFlameThrower(const zeus::CTransform& xf, CStateManager& mgr, float dt)
|
|
|
|
{
|
|
|
|
DeleteFlameThrower(mgr);
|
|
|
|
if (x6e_flameThrowerId != kInvalidUniqueId)
|
|
|
|
return;
|
|
|
|
|
|
|
|
CAssetId resInfo[] =
|
|
|
|
{
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTMainFire"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTMainSmoke"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTSwooshCenter"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTSwooshFire"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTSecondarySmoke"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTSecondaryFire"),
|
|
|
|
NWeaponTypes::get_asset_id_from_name("NFTSecondarySparks"),
|
|
|
|
{}
|
|
|
|
};
|
|
|
|
x6e_flameThrowerId = mgr.AllocateUniqueId();
|
|
|
|
CNewFlameThrower* ft = new CNewFlameThrower(x28_combos[3], "Player_FlameThrower", EWeaponType::Plasma,
|
|
|
|
resInfo, xf, EMaterialTypes::Player, CGunWeapon::GetShotDamageInfo(g_tweakPlayerGun->GetComboShotInfo(3), mgr),
|
2018-02-12 05:30:21 +00:00
|
|
|
x6e_flameThrowerId, kInvalidAreaId, x6c_playerId, EProjectileAttrib::None);
|
2017-09-10 00:36:21 +00:00
|
|
|
mgr.AddObject(ft);
|
|
|
|
ft->Think(dt, mgr);
|
|
|
|
ft->StartFiring(xf, mgr);
|
2017-09-11 02:18:49 +00:00
|
|
|
x24_muzzleFxGen = std::make_unique<CElementGen>(xc_flameMuzzle);
|
2018-09-03 00:46:16 +00:00
|
|
|
x7c_comboSfx = NWeaponTypes::play_sfx(SFXwpn_combo_flamethrower, false, true, 0.165f);
|
2017-09-10 00:36:21 +00:00
|
|
|
mgr.GetCameraManager()->AddCameraShaker(skHardShake, false);
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(true);
|
|
|
|
x74_firingBeamId = CPlayerState::EBeamId::Plasma;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CAuxWeapon::DeleteWaveBusterBeam(CStateManager& mgr)
|
|
|
|
{
|
|
|
|
FreeComboVoiceId();
|
|
|
|
if (x70_waveBusterId != kInvalidUniqueId)
|
|
|
|
{
|
|
|
|
mgr.FreeScriptObject(x70_waveBusterId);
|
|
|
|
x70_waveBusterId = kInvalidUniqueId;
|
|
|
|
x74_firingBeamId = CPlayerState::EBeamId::Invalid;
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-12 05:30:21 +00:00
|
|
|
void CAuxWeapon::CreateWaveBusterBeam(EProjectileAttrib attribs, TUniqueId homingTarget,
|
2017-09-10 00:36:21 +00:00
|
|
|
const zeus::CTransform& xf, CStateManager& mgr)
|
|
|
|
{
|
|
|
|
DeleteFlameThrower(mgr);
|
|
|
|
if (x70_waveBusterId != kInvalidUniqueId)
|
|
|
|
return;
|
|
|
|
|
|
|
|
x70_waveBusterId = mgr.AllocateUniqueId();
|
|
|
|
CWaveBuster* wb = new CWaveBuster(x28_combos[2], EWeaponType::Wave, xf, EMaterialTypes::Player,
|
|
|
|
CGunWeapon::GetShotDamageInfo(g_tweakPlayerGun->GetComboShotInfo(2), mgr), x70_waveBusterId,
|
|
|
|
kInvalidAreaId, x6c_playerId, homingTarget, attribs);
|
|
|
|
mgr.AddObject(wb);
|
2017-09-11 02:18:49 +00:00
|
|
|
x24_muzzleFxGen = std::make_unique<CElementGen>(x18_busterMuzzle);
|
2018-09-03 00:46:16 +00:00
|
|
|
x7c_comboSfx = NWeaponTypes::play_sfx(SFXwpn_combo_wavebuster, false, true, 0.165f);
|
2017-09-10 00:36:21 +00:00
|
|
|
mgr.GetCameraManager()->AddCameraShaker(CCameraShakeData::skChargedShotCameraShakeData, false);
|
|
|
|
mgr.GetPlayerState()->SetFiringComboBeam(true);
|
|
|
|
x74_firingBeamId = CPlayerState::EBeamId::Wave;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const u16 skSoundId[] = { 1810, 1837, 1847, 1842, 1810 };
|
|
|
|
|
|
|
|
void CAuxWeapon::LaunchMissile(float dt, bool underwater, bool charged, CPlayerState::EBeamId currentBeam,
|
2018-02-12 05:30:21 +00:00
|
|
|
EProjectileAttrib attrib, const zeus::CTransform& xf, TUniqueId homingId,
|
2017-09-10 00:36:21 +00:00
|
|
|
CStateManager& mgr)
|
|
|
|
{
|
|
|
|
const SShotParam& info =
|
|
|
|
charged ? g_tweakPlayerGun->GetComboShotInfo(int(currentBeam)) : g_tweakPlayerGun->GetMissileInfo();
|
2018-09-03 00:46:16 +00:00
|
|
|
u16 sfxId = charged ? skSoundId[int(currentBeam)] : u16(SFXwpn_fire_missile);
|
2017-09-10 00:36:21 +00:00
|
|
|
CEnergyProjectile* proj = new CEnergyProjectile(true, charged ? x28_combos[int(currentBeam)] : x0_missile,
|
|
|
|
charged ? EWeaponType::Power : EWeaponType::Missile, xf, EMaterialTypes::Player,
|
|
|
|
CGunWeapon::GetShotDamageInfo(info, mgr), mgr.AllocateUniqueId(), kInvalidAreaId, x6c_playerId, homingId,
|
2018-02-12 05:30:21 +00:00
|
|
|
attrib | EProjectileAttrib::ArmCannon, underwater, zeus::CVector3f::skOne, {}, -1, false);
|
2017-09-10 00:36:21 +00:00
|
|
|
mgr.AddObject(proj);
|
|
|
|
proj->Think(dt, mgr);
|
|
|
|
if (charged)
|
|
|
|
{
|
|
|
|
proj->SetCameraShake(CCameraShakeData::BuildMissileCameraShake(0.25f, 0.75f, 50.f, proj->GetTranslation()));
|
|
|
|
mgr.GetCameraManager()->AddCameraShaker(skHardShake, false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-01-26 09:48:42 +00:00
|
|
|
mgr.GetRumbleManager().Rumble(mgr, ERumbleFxId::PlayerMissileFire, 0.5f, ERumblePriority::One);
|
2017-09-10 00:36:21 +00:00
|
|
|
}
|
|
|
|
x7c_comboSfx = NWeaponTypes::play_sfx(sfxId, underwater, false, 0.165f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CAuxWeapon::Fire(float dt, bool underwater, CPlayerState::EBeamId currentBeam, EChargeState chargeState,
|
2017-09-02 04:06:05 +00:00
|
|
|
const zeus::CTransform& xf, CStateManager& mgr, EWeaponType type, TUniqueId homingId)
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
if (!x80_24_isLoaded)
|
|
|
|
return;
|
2017-09-02 04:06:05 +00:00
|
|
|
|
2018-02-12 05:30:21 +00:00
|
|
|
EProjectileAttrib attrib = EProjectileAttrib::None;
|
2017-09-10 00:36:21 +00:00
|
|
|
if (chargeState == EChargeState::Charged)
|
2018-02-12 05:30:21 +00:00
|
|
|
attrib = CGameProjectile::GetBeamAttribType(type) | EProjectileAttrib::ComboShot;
|
2017-09-10 00:36:21 +00:00
|
|
|
|
|
|
|
if (chargeState == EChargeState::Normal)
|
|
|
|
{
|
|
|
|
LaunchMissile(dt, underwater, chargeState == EChargeState::Charged, currentBeam, attrib, xf, homingId, mgr);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (currentBeam)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Power:
|
|
|
|
case CPlayerState::EBeamId::Ice:
|
|
|
|
LaunchMissile(dt, underwater, chargeState == EChargeState::Charged, currentBeam, attrib, xf, homingId, mgr);
|
|
|
|
break;
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
CreateWaveBusterBeam(attrib, homingId, xf, mgr);
|
|
|
|
break;
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
CreateFlameThrower(xf, mgr, dt);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-09-02 04:06:05 +00:00
|
|
|
}
|
|
|
|
|
2017-08-26 04:36:25 +00:00
|
|
|
void CAuxWeapon::LoadIdle()
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
x80_24_isLoaded = x28_combos[int(x78_loadBeamId)].IsLoaded();
|
2017-08-26 04:36:25 +00:00
|
|
|
}
|
|
|
|
|
2017-08-31 02:42:37 +00:00
|
|
|
void CAuxWeapon::RenderMuzzleFx() const
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
switch (x74_firingBeamId)
|
|
|
|
{
|
|
|
|
case CPlayerState::EBeamId::Wave:
|
|
|
|
case CPlayerState::EBeamId::Plasma:
|
|
|
|
x24_muzzleFxGen->Render();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2017-08-31 02:42:37 +00:00
|
|
|
}
|
|
|
|
|
2017-09-02 04:06:05 +00:00
|
|
|
TUniqueId CAuxWeapon::HasTarget(const CStateManager& mgr) const
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
if (x74_firingBeamId == CPlayerState::EBeamId::Wave)
|
|
|
|
if (auto* wb = static_cast<const CWaveBuster*>(mgr.GetObjectById(x70_waveBusterId)))
|
|
|
|
return wb->GetHomingTargetId();
|
|
|
|
return kInvalidUniqueId;
|
2017-09-02 04:06:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAuxWeapon::SetNewTarget(TUniqueId targetId, CStateManager& mgr)
|
|
|
|
{
|
2017-09-10 00:36:21 +00:00
|
|
|
if (x74_firingBeamId == CPlayerState::EBeamId::Wave)
|
|
|
|
if (auto* wb = static_cast<CWaveBuster*>(mgr.ObjectById(x70_waveBusterId)))
|
|
|
|
wb->SetNewTarget(targetId);
|
2017-09-02 04:06:05 +00:00
|
|
|
}
|
|
|
|
|
2017-08-21 05:46:59 +00:00
|
|
|
}
|