diff --git a/data-canary/monster/magicals/guzzlemaw.lua b/data-canary/monster/magicals/guzzlemaw.lua
index 8e38d459779..391a85118bc 100644
--- a/data-canary/monster/magicals/guzzlemaw.lua
+++ b/data-canary/monster/magicals/guzzlemaw.lua
@@ -23,8 +23,7 @@ monster.Bestiary = {
CharmsPoints = 50,
Stars = 4,
Occurrence = 0,
- Locations = "Guzzlemaw Valley, and a single spawn in a tower in Upper Roshamuul \z
- (south of the Depot and west of the entrance to Roshamuul Prison).",
+ Locations = "Guzzlemaw Valley, and a single spawn in a tower in Upper Roshamuul (south of the Depot and west of the entrance to Roshamuul Prison).",
}
monster.health = 6400
@@ -64,7 +63,6 @@ monster.flags = {
canWalkOnEnergy = true,
canWalkOnFire = true,
canWalkOnPoison = true,
- pet = false,
}
monster.light = {
@@ -78,63 +76,64 @@ monster.voices = {
}
monster.loot = {
- { id = 3031, chance = 100000, maxCount = 100 }, -- gold coin
- { id = 3035, chance = 100000, maxCount = 7 }, -- platinum coin
- { id = 3104, chance = 10700 }, -- banana skin
- { id = 3110, chance = 10500 }, -- piece of iron
- { id = 3111, chance = 9500 }, -- fishbone
+ { name = "gold coin", chance = 100000, maxCount = 100 },
+ { name = "platinum coin", chance = 100000, maxCount = 7 },
+ { name = "banana skin", chance = 10700 },
+ { name = "piece of iron", chance = 10500 },
+ { name = "fishbone", chance = 9500 },
{ id = 3114, chance = 10400 }, -- skull
{ id = 3115, chance = 9200 }, -- bone
{ id = 3116, chance = 4500 }, -- big bone
- { id = 3265, chance = 2700 }, -- two handed sword
+ { name = "two handed sword", chance = 2700 },
{ id = 3578, chance = 7000, maxCount = 3 }, -- fish
- { id = 3582, chance = 10000 }, -- ham
- { id = 5880, chance = 3000 }, -- iron ore
- { id = 5895, chance = 5000 }, -- fish fin
- { id = 5925, chance = 5700 }, -- hardened bone
+ { name = "ham", chance = 10000 },
+ { name = "iron ore", chance = 3000 },
+ { name = "fish fin", chance = 5000 },
+ { name = "hardened bone", chance = 5700 },
{ id = 5951, chance = 9400 }, -- fish tail
- { id = 7404, chance = 1000 }, -- assassin dagger
- { id = 7407, chance = 2000 }, -- haunted blade
- { id = 7418, chance = 380 }, -- nightmare blade
- { id = 238, chance = 17000, maxCount = 3 }, -- great mana potion
- { id = 239, chance = 18500, maxCount = 2 }, -- great health potion
- { id = 10389, chance = 1200 }, -- sai
- { id = 16120, chance = 3000 }, -- violet crystal shard
- { id = 16123, chance = 12000, maxCount = 2 }, -- brown crystal splinter
- { id = 16126, chance = 7600 }, -- red crystal fragment
+ { name = "assassin dagger", chance = 1000 },
+ { name = "haunted blade", chance = 2000 },
+ { name = "nightmare blade", chance = 380 },
+ { name = "great mana potion", chance = 17000, maxCount = 3 },
+ { name = "great health potion", chance = 18500, maxCount = 2 },
+ { name = "sai", chance = 1200 },
+ { name = "violet crystal shard", chance = 3000 },
+ { name = "brown crystal splinter", chance = 12000, maxCount = 2 },
+ { name = "red crystal fragment", chance = 7600 },
{ id = 16279, chance = 12000 }, -- crystal rubbish
- { id = 20062, chance = 920 }, -- cluster of solace
- { id = 20198, chance = 15000 }, -- frazzle tongue
- { id = 20199, chance = 14000 }, -- frazzle skin
+ { name = "cluster of solace", chance = 8920 },
+ { name = "frazzle tongue", chance = 15000 },
+ { name = "frazzle skin", chance = 14000 },
}
monster.attacks = {
{ name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -499 },
-- bleed
- { name = "condition", type = CONDITION_BLEEDING, interval = 2000, chance = 10, minDamage = -500, maxDamage = -1000, radius = 3, target = false },
- { name = "combat", interval = 2000, chance = 10, type = COMBAT_PHYSICALDAMAGE, minDamage = 0, maxDamage = -900, length = 8, spread = 3, effect = CONST_ME_EXPLOSIONAREA, target = false },
+ { name = "condition", type = CONDITION_BLEEDING, interval = 2000, chance = 10, minDamage = -500, maxDamage = -1000, radius = 3, effect = CONST_ME_DRAWBLOOD, target = false },
+ { name = "combat", interval = 2000, chance = 10, type = COMBAT_LIFEDRAIN, minDamage = 0, maxDamage = -900, length = 8, spread = 0, effect = CONST_ME_EXPLOSIONAREA, target = false },
{ name = "combat", interval = 2000, chance = 20, type = COMBAT_PHYSICALDAMAGE, minDamage = 0, maxDamage = -500, radius = 2, shootEffect = CONST_ANI_LARGEROCK, effect = CONST_ME_STONES, target = true },
- { name = "speed", interval = 2000, chance = 15, speedChange = -100, radius = 6, effect = CONST_ME_MAGIC_RED, target = false, duration = 15000 },
- { name = "combat", interval = 2000, chance = 10, type = COMBAT_LIFEDRAIN, minDamage = 0, maxDamage = -800, length = 8, spread = 3, effect = CONST_ME_MAGIC_RED, target = false },
+ { name = "speed", interval = 2000, chance = 15, speedChange = -800, radius = 6, effect = CONST_ME_MAGIC_RED, target = false, duration = 15000 },
+ { name = "combat", interval = 2000, chance = 10, type = COMBAT_LIFEDRAIN, minDamage = 0, maxDamage = -800, length = 8, spread = 0, effect = CONST_ME_MAGIC_RED, target = false },
}
monster.defenses = {
defense = 50,
- armor = 50,
+ armor = 74,
+ mitigation = 2.31,
{ name = "combat", interval = 2000, chance = 20, type = COMBAT_HEALING, minDamage = 250, maxDamage = 425, effect = CONST_ME_HITBYPOISON, target = false },
}
monster.elements = {
- { type = COMBAT_PHYSICALDAMAGE, percent = 10 },
- { type = COMBAT_ENERGYDAMAGE, percent = 5 },
- { type = COMBAT_EARTHDAMAGE, percent = 15 },
- { type = COMBAT_FIREDAMAGE, percent = 5 },
+ { type = COMBAT_PHYSICALDAMAGE, percent = 5 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 15 },
+ { type = COMBAT_EARTHDAMAGE, percent = 20 },
+ { type = COMBAT_FIREDAMAGE, percent = 10 },
{ type = COMBAT_LIFEDRAIN, percent = 0 },
{ type = COMBAT_MANADRAIN, percent = 0 },
{ type = COMBAT_DROWNDAMAGE, percent = 0 },
{ type = COMBAT_ICEDAMAGE, percent = 5 },
- { type = COMBAT_HOLYDAMAGE, percent = 0 },
- { type = COMBAT_DEATHDAMAGE, percent = 5 },
+ { type = COMBAT_HOLYDAMAGE, percent = -5 },
+ { type = COMBAT_DEATHDAMAGE, percent = 10 },
}
monster.immunities = {
diff --git a/data-canary/scripts/actions/tools/kitchen_knife.lua b/data-canary/scripts/actions/tools/kitchen_knife.lua
deleted file mode 100644
index 921baf9092d..00000000000
--- a/data-canary/scripts/actions/tools/kitchen_knife.lua
+++ /dev/null
@@ -1,8 +0,0 @@
-local kitchenKnife = Action()
-
-function kitchenKnife.onUse(player, item, fromPosition, target, toPosition, isHotkey)
- return ActionsLib.useKitchenKnife(player, item, fromPosition, target, toPosition, isHotkey)
-end
-
-kitchenKnife:id(3469)
-kitchenKnife:register()
diff --git a/data-canary/scripts/actions/tools/sickle.lua b/data-canary/scripts/actions/tools/sickle.lua
deleted file mode 100644
index a4e024dde20..00000000000
--- a/data-canary/scripts/actions/tools/sickle.lua
+++ /dev/null
@@ -1,8 +0,0 @@
-local sickle = Action()
-
-function sickle.onUse(player, item, fromPosition, target, toPosition, isHotkey)
- return ActionsLib.useSickle(player, item, fromPosition, target, toPosition, isHotkey) or ActionsLib.destroyItem(player, target, toPosition)
-end
-
-sickle:id(3293, 3306, 32595)
-sickle:register()
diff --git a/data-canary/scripts/lib/register_actions.lua b/data-canary/scripts/lib/register_actions.lua
index 7c0aa49102e..e59a46b3b03 100644
--- a/data-canary/scripts/lib/register_actions.lua
+++ b/data-canary/scripts/lib/register_actions.lua
@@ -1,12 +1,17 @@
-local holeId = { 294, 369, 370, 385, 394, 411, 412, 413, 432, 433, 435, 8709, 594, 595, 615, 609, 610, 615, 1156, 482, 483, 868, 874, 4824, 7768, 433, 432, 413, 7767, 411, 370, 369, 7737, 7755, 7768, 7767, 7515, 7516, 7517, 7518, 7519, 7520, 7521, 7522, 7762, 8144, 8690, 8709, 12203, 12961, 17239, 19220, 23364 } -- usable rope holes, for rope spots see global.lua
-local wildGrowth = { 2130, 2130, 2982, 2524, 2030, 2029, 10182 } -- wild growth destroyable by machete
-local jungleGrass = { [3696] = 3695, [3702] = 3701, [17153] = 17151 } -- grass destroyable by machete
-local groundIds = { 354, 355 } -- pick usable ground
+local holeId = { 294, 369, 370, 385, 394, 411, 412, 413, 432, 433, 435, 482, 483, 594, 595, 609, 610, 615, 868, 874, 1156, 4824, 7515, 7516, 7517, 7518, 7519, 7520, 7521, 7522, 7737, 7755, 7762, 7767, 7768, 8144, 8690, 8709, 12203, 12961, 17239, 19220, 23364 } -- usable rope holes, for rope spots see global.lua
+local wildGrowth = { 2130, 3635, 30224 } -- wild growth destroyable by machete
+local jungleGrass = { -- grass destroyable by machete
+ [3696] = 3695,
+ [3702] = 3701,
+ [17153] = 17151,
+}
+local groundIds = { 354, 355 } -- pick usable grounds
local sandIds = { 231 } -- desert sand
-local fruits = { 2673, 2674, 2675, 2676, 2677, 2678, 2679, 2680, 2681, 2682, 2684, 2685, 5097, 8839, 8840, 8841 } -- fruits to make decorated cake with knife
+local fruits = { 3584, 3585, 3586, 3587, 3588, 3589, 3590, 3591, 3592, 3593, 3595, 3596, 5096, 8011, 8012, 8013 } -- fruits to make decorated cake with knife
local holes = { 593, 606, 608, 867, 21341 } -- holes opened by shovel
+local ropeSpots = { 386, 421, 12935, 12936, 14238, 17238, 21501, 21965, 21966, 21967, 21968, 23363 }
-function destroyItem(player, target, toPosition)
+function destroyItem(player, item, fromPosition, target, toPosition, isHotkey)
if type(target) ~= "userdata" or not target:isItem() then
return false
end
@@ -40,10 +45,8 @@ function destroyItem(player, target, toPosition)
end
end
end
-
target:remove(1)
end
-
toPosition:sendMagicEffect(CONST_ME_POFF)
return true
end
@@ -67,7 +70,7 @@ function onUseMachete(player, item, fromPosition, target, toPosition, isHotkey)
player:addAchievementProgress("Nothing Can Stop Me", 100)
return true
end
- return destroyItem(player, target, toPosition)
+ return destroyItem(player, item, fromPosition, target, toPosition, isHotkey)
end
function onUsePick(player, item, fromPosition, target, toPosition, isHotkey)
@@ -82,7 +85,6 @@ function onUsePick(player, item, fromPosition, target, toPosition, isHotkey)
else
player:addItem(3028) -- 49% chance of getting small diamond
end
-
player:addAchievementProgress("Petrologist", 100)
target:getPosition():sendMagicEffect(CONST_ME_BLOCKHIT)
target:remove(1)
@@ -99,7 +101,7 @@ function onUsePick(player, item, fromPosition, target, toPosition, isHotkey)
return false
end
- if table.contains(groundIds, ground.itemid) and ground.actionid == actionIds.pickHole then
+ if table.contains(groundIds, ground.itemid) and (ground:hasAttribute(ITEM_ATTRIBUTE_UNIQUEID) or ground:hasAttribute(ITEM_ATTRIBUTE_ACTIONID)) then
ground:transform(394)
ground:decay()
toPosition:sendMagicEffect(CONST_ME_POFF)
@@ -127,41 +129,28 @@ function onUseRope(player, item, fromPosition, target, toPosition, isHotkey)
local ground = tile:getGround()
if ground and table.contains(ropeSpots, ground:getId()) or tile:getItemById(12935) then
- tile = Tile(toPosition:moveUpstairs())
- if not tile then
- return false
- end
-
- if tile:hasFlag(TILESTATE_PROTECTIONZONE) and player:isPzLocked() then
+ if Tile(toPosition:moveUpstairs()):hasFlag(TILESTATE_PROTECTIONZONE) and player:isPzLocked() then
player:sendCancelMessage(RETURNVALUE_PLAYERISPZLOCKED)
return true
end
-
player:teleportTo(toPosition, false)
return true
- end
-
- if table.contains(holeId, target.itemid) then
+ elseif table.contains(holeId, target.itemid) then
toPosition.z = toPosition.z + 1
tile = Tile(toPosition)
- if not tile then
- return false
- end
-
- local thing = tile:getTopVisibleThing()
- if not thing then
- return true
- end
-
- if thing:isPlayer() then
- if Tile(toPosition:moveUpstairs()):queryAdd(thing) ~= RETURNVALUE_NOERROR then
- return false
+ if tile then
+ local thing = tile:getTopVisibleThing()
+ if thing:isPlayer() then
+ if Tile(toPosition:moveUpstairs()):hasFlag(TILESTATE_PROTECTIONZONE) and thing:isPzLocked() then
+ return false
+ end
+ return thing:teleportTo(toPosition, false)
+ end
+ if thing:isItem() and thing:getType():isMovable() then
+ return thing:moveTo(toPosition:moveUpstairs())
end
-
- return thing:teleportTo(toPosition, false)
- elseif thing:isItem() and thing:getType():isMovable() then
- return thing:moveTo(toPosition:moveUpstairs())
end
+ player:sendCancelMessage(RETURNVALUE_NOTPOSSIBLE)
return true
end
return false
@@ -182,7 +171,13 @@ function onUseShovel(player, item, fromPosition, target, toPosition, isHotkey)
if table.contains(holes, groundId) then
ground:transform(groundId + 1)
ground:decay()
-
+ toPosition:moveDownstairs()
+ toPosition.y = toPosition.y - 1
+ if Tile(toPosition):hasFlag(TILESTATE_PROTECTIONZONE) and player:isPzLocked() then
+ player:sendCancelMessage(RETURNVALUE_PLAYERISPZLOCKED)
+ return true
+ end
+ player:teleportTo(toPosition, false)
toPosition.z = toPosition.z + 1
tile:relocateTo(toPosition)
player:addAchievementProgress("The Undertaker", 500)
@@ -192,14 +187,14 @@ function onUseShovel(player, item, fromPosition, target, toPosition, isHotkey)
player:addAchievementProgress("The Undertaker", 500)
elseif target.itemid == 17950 then -- swamp digging
if not player:hasExhaustion("swamp-digging") then
- local chance = math.random(100)
- if chance >= 1 and chance <= 42 then
+ local chance = math.random(1, 100)
+ if chance <= 42 then
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You dug up a dead snake.")
player:addItem(4259)
elseif chance >= 43 and chance <= 79 then
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You dug up a small diamond.")
player:addItem(3028)
- elseif chance >= 80 then
+ else
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You dug up a leech.")
player:addItem(17858)
end
@@ -209,16 +204,15 @@ function onUseShovel(player, item, fromPosition, target, toPosition, isHotkey)
end
elseif table.contains(sandIds, groundId) then
local randomValue = math.random(1, 100)
- if target.actionid == actionIds.sandHole and randomValue <= 20 then
+ if target.actionid == 100 and randomValue <= 20 then
ground:transform(615)
ground:decay()
elseif randomValue == 1 then
- Game.createItem(3042, 1, toPosition)
+ Game.createItem(3042, 1, toPosition) -- Scarab Coin
player:addAchievementProgress("Gold Digger", 100)
elseif randomValue > 95 then
Game.createMonster("Scarab", toPosition)
end
-
toPosition:sendMagicEffect(CONST_ME_POFF)
else
return false
@@ -238,6 +232,13 @@ function onUseScythe(player, item, fromPosition, target, toPosition, isHotkey)
player:addAchievementProgress("Happy Farmer", 200)
return true
end
+ return destroyItem(player, item, fromPosition, target, toPosition, isHotkey)
+end
+
+function onUseSickle(player, item, fromPosition, target, toPosition, isHotkey)
+ if not table.contains({ 3293, 3306 }, item.itemid) then
+ return false
+ end
if target.itemid == 5464 then -- burning sugar cane
target:transform(5463)
@@ -246,14 +247,14 @@ function onUseScythe(player, item, fromPosition, target, toPosition, isHotkey)
player:addAchievementProgress("Natural Sweetener", 50)
return true
end
- return destroyItem(player, target, toPosition)
+ return destroyItem(player, item, fromPosition, target, toPosition, isHotkey)
end
function onUseCrowbar(player, item, fromPosition, target, toPosition, isHotkey)
if not table.contains({ 3304, 9598 }, item.itemid) then
return false
end
- return destroyItem(player, target, toPosition)
+ return destroyItem(player, item, fromPosition, target, toPosition, isHotkey)
end
function onUseKitchenKnife(player, item, fromPosition, target, toPosition, isHotkey)
@@ -269,3 +270,10 @@ function onUseKitchenKnife(player, item, fromPosition, target, toPosition, isHot
end
return false
end
+
+function onUseSpoon(player, item, fromPosition, target, toPosition, isHotkey)
+ if not table.contains({ 3468, 3470 }, item.itemid) then
+ return false
+ end
+ return false
+end
diff --git a/data-canary/scripts/spells/monster/fury_skill_reducer.lua b/data-canary/scripts/spells/monster/fury_skill_reducer.lua
new file mode 100644
index 00000000000..1e8acd0ae7a
--- /dev/null
+++ b/data-canary/scripts/spells/monster/fury_skill_reducer.lua
@@ -0,0 +1,27 @@
+local combat = {}
+
+for i = 65, 80 do
+ combat[i] = Combat()
+ combat[i]:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_SOUND_YELLOW)
+
+ local condition = Condition(CONDITION_ATTRIBUTES)
+ condition:setParameter(CONDITION_PARAM_TICKS, 5000)
+ condition:setParameter(CONDITION_PARAM_SKILL_DEFENSEPERCENT, i)
+
+ local area = createCombatArea(AREA_CIRCLE3X3)
+ combat[i]:setArea(area)
+ combat[i]:addCondition(condition)
+end
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat[math.random(65, 80)]:execute(creature, var)
+end
+
+spell:name("fury skill reducer")
+spell:words("###2")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:register()
diff --git a/data-canary/world/canary-house.xml b/data-canary/world/canary-house.xml
index adaa1a1c8f2..43d8e80d11e 100644
--- a/data-canary/world/canary-house.xml
+++ b/data-canary/world/canary-house.xml
@@ -1,4 +1,4 @@
-
+
diff --git a/data-canary/world/canary-zones.xml b/data-canary/world/canary-zones.xml
new file mode 100644
index 00000000000..a9224bd3c2d
--- /dev/null
+++ b/data-canary/world/canary-zones.xml
@@ -0,0 +1,2 @@
+
+
diff --git a/data-canary/world/canary.otbm b/data-canary/world/canary.otbm
index 280d57a9194..8f787811a53 100644
Binary files a/data-canary/world/canary.otbm and b/data-canary/world/canary.otbm differ
diff --git a/data-otservbr-global/lib/quests/quest.lua b/data-otservbr-global/lib/quests/quest.lua
index 21b1744fec6..392367c9d94 100644
--- a/data-otservbr-global/lib/quests/quest.lua
+++ b/data-otservbr-global/lib/quests/quest.lua
@@ -5,5 +5,6 @@ dofile(DATA_DIRECTORY .. "/lib/quests/killing_in_the_name_of.lua")
dofile(DATA_DIRECTORY .. "/lib/quests/svargrond_arena.lua")
dofile(DATA_DIRECTORY .. "/lib/quests/the_cursed_crystal.lua")
dofile(DATA_DIRECTORY .. "/lib/quests/the_queen_of_the_banshees.lua")
+dofile(DATA_DIRECTORY .. "/lib/quests/soul_war.lua")
dofile(DATA_DIRECTORY .. "/lib/quests/their_masters_voice.lua")
dofile(DATA_DIRECTORY .. "/lib/quests/the_primal_ordeal.lua")
diff --git a/data-otservbr-global/lib/quests/soul_war.lua b/data-otservbr-global/lib/quests/soul_war.lua
new file mode 100644
index 00000000000..a9e9d920e91
--- /dev/null
+++ b/data-otservbr-global/lib/quests/soul_war.lua
@@ -0,0 +1,1589 @@
+SoulWarQuest = {
+ -- Item ids
+ -- Goshnar's Hatred
+ bagYouDesireItemId = 34109,
+ goshnarsHatredSorrowId = 33793,
+ condensedRemorseId = 33792,
+ -- Goshnar's Spite
+ weepingSoulCorpseId = 33876,
+ searingFireId = 33877,
+ -- Goshnar's Cruelty
+ pulsatingEnergyId = 34005,
+ greedyMawId = 33890,
+ someMortalEssenceId = 33891,
+ theBloodOfCloakTerrorIds = { 33854, 34006, 34007 },
+ -- Goshnar's Megalomania
+ deadAspectOfPowerCorpseId = 33949,
+ cleansedSanityItemId = 33950,
+ necromanticRemainsItemId = 33984,
+
+ poolDamagePercentages = {
+ [33854] = 0.20, -- 20% of maximum health for the largest pool
+ [34006] = 0.15, -- 15% for a medium-sized pool
+ [34007] = 0.10, -- 10% for the smallest pool
+ },
+
+ timeToIncreaseCrueltyDefense = 15, -- In seconds, it will increase every 15 seconds if don't use mortal essence in greedy maw
+ useGreedMawCooldown = 30, -- In seconds
+ goshnarsCrueltyDefenseChange = 2, -- Defense change, the amount that will decrease or increase defense, the defense cannot decrease more than the monster's original defense amount
+ goshnarsCrueltyWaveInterval = 7, -- In seconds
+
+ timeToReturnImmuneMegalomania = 70, -- In seconds
+
+ bagYouDesireChancePerTaint = 10, -- Increases % per taint
+ bagYouDesireMonsters = {
+ "Bony Sea Devil",
+ "Brachiodemon",
+ "Branchy Crawler",
+ "Capricious Phantom",
+ "Cloak of Terror",
+ "Courage Leech",
+ "Distorted Phantom",
+ "Druid's Apparition",
+ "Infernal Demon",
+ "Infernal Phantom",
+ "Knight's Apparition",
+ "Many Faces",
+ "Mould Phantom",
+ "Paladin's Apparition",
+ "Rotten Golem",
+ "Sorcerer's Apparition",
+ "Turbulent Elemental",
+ "Vibrant Phantom",
+ "Hazardous Phantom",
+ "Goshnar's Cruelty",
+ "Goshnar's Spite",
+ "Goshnar's Malice",
+ "Goshnar's Hatred",
+ "Goshnar's Greed",
+ "Goshnar's Megalomania",
+ },
+
+ -- Goshnar's Cruelty pulsating energy monsters
+ pulsatingEnergyMonsters = {
+ "Vibrant Phantom",
+ "Cloak of Terror",
+ "Courage Leech",
+ },
+
+ miniBosses = {
+ ["Goshnar's Malice"] = true,
+ ["Goshnar's Hatred"] = true,
+ ["Goshnar's Spite"] = true,
+ ["Goshnar's Cruelty"] = true,
+ ["Goshnar's Greed"] = true,
+ },
+
+ finalRewards = {
+ { id = 34082, name = "soulcutter" },
+ { id = 34083, name = "soulshredder" },
+ { id = 34084, name = "soulbiter" },
+ { id = 34085, name = "souleater" },
+ { id = 34086, name = "soulcrusher" },
+ { id = 34087, name = "soulmaimer" },
+ { id = 34088, name = "soulbleeder" },
+ { id = 34089, name = "soulpiercer" },
+ { id = 34090, name = "soultainter" },
+ { id = 34091, name = "soulhexer" },
+ { id = 34092, name = "soulshanks" },
+ { id = 34093, name = "soulstrider" },
+ { id = 34094, name = "soulshell" },
+ { id = 34095, name = "soulmantel" },
+ { id = 34096, name = "soulshroud" },
+ { id = 34097, name = "pair of soulwalkers" },
+ { id = 34098, name = "pair of soulstalkers" },
+ { id = 34099, name = "soulbastion" },
+ },
+
+ kvSoulWar = KV.scoped("quest"):scoped("soul-war"),
+ -- Global KV for storage burning change form time
+ kvBurning = KV.scoped("quest"):scoped("soul-war"):scoped("burning-change-form"),
+
+ rottenWastelandShrines = {
+ [33019] = { x = 33926, y = 31091, z = 13 },
+ [33021] = { x = 33963, y = 31078, z = 13 },
+ [33022] = { x = 33970, y = 30988, z = 13 },
+ [33024] = { x = 33970, y = 31012, z = 13 },
+ },
+
+ -- Lever room and teleports positions
+ goshnarsGreedAccessPosition = { from = { x = 33937, y = 31217, z = 11 }, to = { x = 33782, y = 31665, z = 14 } },
+ goshnarsHatredAccessPosition = { from = { x = 33914, y = 31032, z = 12 }, to = { x = 33774, y = 31604, z = 14 } },
+ -- Teleports from 1st/2nd/3rd floors
+ goshnarsCrueltyTeleportRoomPositions = {
+ { from = Position(33889, 31873, 3), to = Position(33830, 31881, 4), access = "first-floor-access", count = 40 },
+ { from = Position(33829, 31880, 4), to = Position(33856, 31889, 5), access = "second-floor-access", count = 55 },
+ { from = Position(33856, 31884, 5), to = Position(33857, 31865, 6), access = "third-floor-access", count = 70 },
+ },
+
+ claustrophobicInfernoRaids = {
+ [1] = {
+ zoneArea = {
+ { x = 33985, y = 31053, z = 9 },
+ { x = 34045, y = 31077, z = 9 },
+ },
+ sandTimerPositions = {
+ { x = 34012, y = 31049, z = 9 },
+ { x = 34013, y = 31049, z = 9 },
+ { x = 34014, y = 31049, z = 9 },
+ { x = 34015, y = 31049, z = 9 },
+ },
+ zone = Zone("raid.first-claustrophobic-inferno"),
+ spawns = {
+ Position(33991, 31064, 9),
+ Position(34034, 31060, 9),
+ Position(34028, 31067, 9),
+ Position(34020, 31067, 9),
+ Position(34008, 31067, 9),
+ Position(34001, 31059, 9),
+ Position(33992, 31069, 9),
+ Position(34002, 31072, 9),
+ Position(34013, 31074, 9),
+ Position(33998, 31060, 9),
+ Position(34039, 31065, 9),
+ Position(34032, 31072, 9),
+ },
+ exitPosition = { x = 34009, y = 31083, z = 9 },
+ getZone = function()
+ return SoulWarQuest.claustrophobicInfernoRaids[1].zone
+ end,
+ },
+ [2] = {
+ zoneArea = {
+ { x = 33988, y = 31042, z = 10 },
+ { x = 34043, y = 31068, z = 10 },
+ },
+ sandTimerPositions = {
+ { x = 34012, y = 31075, z = 10 },
+ { x = 34011, y = 31075, z = 10 },
+ { x = 34010, y = 31075, z = 10 },
+ },
+ zone = Zone("raid.second-claustrophobic-inferno"),
+ spawns = {
+ Position(33999, 31046, 10),
+ Position(34011, 31047, 10),
+ Position(34015, 31052, 10),
+ Position(34021, 31044, 10),
+ Position(34029, 31054, 10),
+ Position(34037, 31052, 10),
+ Position(34037, 31060, 10),
+ Position(34023, 31062, 10),
+ Position(34012, 31061, 10),
+ Position(33998, 31061, 10),
+ Position(34005, 31052, 10),
+ },
+ exitPosition = { x = 34011, y = 31028, z = 10 },
+ getZone = function()
+ return SoulWarQuest.claustrophobicInfernoRaids[2].zone
+ end,
+ },
+ [3] = {
+ zoneArea = {
+ { x = 33987, y = 31043, z = 11 },
+ { x = 34044, y = 31076, z = 11 },
+ },
+ sandTimerPositions = {
+ { x = 34009, y = 31036, z = 11 },
+ { x = 34010, y = 31036, z = 11 },
+ { x = 34011, y = 31036, z = 11 },
+ { x = 34012, y = 31036, z = 11 },
+ { x = 34013, y = 31036, z = 11 },
+ { x = 34014, y = 31036, z = 11 },
+ },
+ zone = Zone("raid.third-claustrophobic-inferno"),
+ spawns = {
+ Position(34005, 31049, 11),
+ Position(33999, 31051, 11),
+ Position(33995, 31055, 11),
+ Position(33999, 31068, 11),
+ Position(34016, 31068, 11),
+ Position(34030, 31070, 11),
+ Position(34038, 31066, 11),
+ Position(34038, 31051, 11),
+ Position(34033, 31051, 11),
+ Position(34025, 31049, 11),
+ Position(34013, 31058, 11),
+ Position(34021, 31059, 11),
+ Position(34027, 31063, 11),
+ Position(34007, 31063, 11),
+ Position(34004, 31059, 11),
+ },
+ exitPosition = { x = 34014, y = 31085, z = 11 },
+ getZone = function()
+ return SoulWarQuest.claustrophobicInfernoRaids[3].zone
+ end,
+ },
+ spawnTime = 10, -- seconds
+ suriviveTime = 2 * 60, -- 2 minutes
+ timeToKick = 5, -- seconds
+ },
+
+ areaZones = {
+ monsters = {
+ ["zone.claustrophobic-inferno"] = "Brachiodemon",
+ ["zone.mirrored-nightmare"] = "Many Faces",
+ ["zone.ebb-and-flow"] = "Bony Sea Devil",
+ ["zone.furious-crater"] = "Cloak of Terror",
+ ["zone.rotten-wasteland"] = "Branchy Crawler",
+ ["boss.goshnar's-malice"] = "Dreadful Harvester",
+ ["boss.goshnar's-spite"] = "Dreadful Harvester",
+ ["boss.goshnar's-greed"] = "Dreadful Harvester",
+ ["boss.goshnar's-hatred"] = "Dreadful Harvester",
+ ["boss.goshnar's-cruelty"] = "Dreadful Harvester",
+ ["boss.goshnar's-megalomania-purple"] = "Dreadful Harvester",
+ },
+
+ claustrophobicInferno = Zone("zone.claustrophobic-inferno"),
+ mirroredNightmare = Zone("zone.mirrored-nightmare"),
+ ebbAndFlow = Zone("zone.ebb-and-flow"),
+ furiousCrater = Zone("zone.furious-crater"),
+ rottenWasteland = Zone("zone.rotten-wasteland"),
+ },
+
+ -- Levers configuration
+ levers = {
+ goshnarsMalicePosition = { x = 33678, y = 31599, z = 14 },
+ goshnarsSpitePosition = { x = 33773, y = 31634, z = 14 },
+ goshnarsGreedPosition = { x = 33775, y = 31665, z = 14 },
+ goshnarsHatredPosition = { x = 33772, y = 31601, z = 14 },
+ goshnarsCrueltyPosition = { x = 33853, y = 31854, z = 6 },
+ goshnarsMegalomaniaPosition = { x = 33675, y = 31634, z = 14 },
+
+ -- Levers system
+ goshnarsSpite = {
+ boss = {
+ name = "Goshnar's Spite",
+ position = Position(33743, 31632, 14),
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33774, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33775, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33776, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33777, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33778, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33734, 31624, 14),
+ to = Position(33751, 31640, 14),
+ },
+ onUseExtra = function(player)
+ local zone = Zone("boss.goshnar's-spite")
+ if zone then
+ local positions = zone:getPositions()
+ for _, pos in ipairs(positions) do
+ local tile = Tile(pos)
+ if tile then
+ local item = tile:getItemById(SoulWarQuest.weepingSoulCorpseId)
+ if item then
+ logger.debug("Weeping Soul Corpse removed from position: {}", pos)
+ item:remove()
+ end
+ end
+ end
+ end
+ end,
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 20 * 60 * 60, -- 20 hours
+ },
+ goshnarsMalice = {
+ boss = {
+ name = "Goshnar's Malice",
+ position = Position(33709, 31599, 14),
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33679, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33680, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33681, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33682, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33683, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33699, 31590, 14),
+ to = Position(33718, 31607, 14),
+ },
+ onUseExtra = function(player)
+ addEvent(SpawnSoulCage, 23000)
+ end,
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 20 * 60 * 60, -- 20 hours
+ },
+ goshnarsGreed = {
+ boss = {
+ name = "Goshnar's Greed",
+ position = Position(33746, 31666, 14),
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33776, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33777, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33778, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33779, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33780, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33737, 31658, 14),
+ to = Position(33755, 31673, 14),
+ },
+ timeToFightAgain = 0, -- TODO: Remove later
+ onUseExtra = function()
+ CreateGoshnarsGreedMonster("Greedbeast", Position(33744, 31666, 14))
+ CreateGoshnarsGreedMonster("Soulsnatcher", Position(33747, 31668, 14))
+ CreateGoshnarsGreedMonster("Weak Soul", Position(33750, 31666, 14))
+ end,
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 20 * 60 * 60, -- 20 hours
+ },
+ goshnarsHatred = {
+ boss = {
+ name = "Goshnar's Hatred",
+ position = Position(33744, 31599, 14),
+ },
+ monsters = {
+ { name = "Ashes of Burning Hatred", pos = { x = 33743, y = 31599, z = 14 } },
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33773, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33774, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33775, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33776, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33777, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33735, 31592, 14),
+ to = Position(33751, 31606, 14),
+ },
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 20 * 60 * 60, -- 20 hours
+ onUseExtra = function(player)
+ SoulWarQuest.kvBurning:set("time", 180)
+ logger.trace("Goshnar's Hatred burning change form time set to: {}", 180)
+ player:resetGoshnarSymbolTormentCounter()
+ end,
+ },
+ goshnarsCruelty = {
+ boss = {
+ name = "Goshnar's Cruelty",
+ position = Position(33856, 31866, 7),
+ },
+ monsters = {
+ { name = "A Greedy Eye", pos = { x = 33856, y = 31858, z = 7 } },
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33854, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
+ { pos = Position(33855, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
+ { pos = Position(33856, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
+ { pos = Position(33857, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
+ { pos = Position(33858, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33847, 31858, 7),
+ to = Position(33864, 31874, 7),
+ },
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 20 * 60 * 60, -- 20 hours
+ onUseExtra = function(player)
+ SoulWarQuest.kvSoulWar:remove("greedy-maw-action")
+ SoulWarQuest.kvSoulWar:remove("goshnars-cruelty-defense-drain")
+ player:soulWarQuestKV():scoped("furious-crater"):remove("greedy-maw-action")
+ end,
+ },
+ goshnarsMegalomania = {
+ boss = {
+ name = "Goshnar's Megalomania Purple",
+ position = Position(33710, 31634, 14),
+ },
+ monsters = {
+ { name = "Aspect of Power", pos = { x = 33710, y = 31635, z = 14 } },
+ },
+ requiredLevel = 250,
+ playerPositions = {
+ { pos = Position(33676, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33677, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33678, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33679, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
+ { pos = Position(33680, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
+ },
+ specPos = {
+ from = Position(33701, 31626, 14),
+ to = Position(33719, 31642, 14),
+ },
+ exit = Position(33621, 31427, 10),
+ timeToFightAgain = 72 * 60 * 60, -- 72 hours
+ onUseExtra = function(player)
+ player:resetGoshnarSymbolTormentCounter()
+ SoulWarQuest.kvSoulWar:remove("cleansed-sanity-action")
+ player:soulWarQuestKV():scoped("furious-crater"):remove("cleansed-sanity-action")
+ end,
+ },
+ },
+
+ -- Goshnar's Greed
+ apparitionNames = {
+ "Druid's Apparition",
+ "Knight's Apparition",
+ "Paladin's Apparition",
+ "Sorcerer's Apparition",
+ },
+
+ burningTransformations = {
+ { 180, "Ashes of Burning Hatred" },
+ { 135, "Spark of Burning Hatred" },
+ { 90, "Flame of Burning Hatred" },
+ { 45, "Blaze of Burning Hatred" },
+ },
+
+ burningHatredMonsters = {
+ "Ashes of Burning Hatred",
+ "Spark of Burning Hatred",
+ "Flame of Burning Hatred",
+ "Blaze of Burning Hatred",
+ },
+
+ requiredCountPerApparition = 25,
+
+ -- Ebb and flow
+ ebbAndFlow = {
+ zone = Zone("ebb-and-flow-zone"),
+ -- Positions to teleport into rooms when innundate map is loaded
+ centerRoomPositions = {
+ { conor = { x = 33929, y = 31020, z = 9 }, teleportPosition = { x = 33939, y = 31021, z = 8 } },
+ { conor = { x = 33929, y = 31047, z = 9 }, teleportPosition = { x = 33938, y = 31047, z = 8 } },
+ { conor = { x = 33918, y = 31047, z = 9 }, teleportPosition = { x = 33903, y = 31049, z = 8 } },
+ { conor = { x = 33898, y = 31054, z = 9 }, teleportPosition = { x = 33903, y = 31049, z = 8 } },
+ { conor = { x = 33929, y = 31047, z = 9 }, teleportPosition = { x = 33938, y = 31047, z = 8 } },
+ { conor = { x = 33940, y = 31054, z = 9 }, teleportPosition = { x = 33938, y = 31047, z = 8 } },
+ { conor = { x = 33940, y = 31064, z = 9 }, teleportPosition = { x = 33937, y = 31074, z = 8 } },
+ { conor = { x = 33937, y = 31086, z = 9 }, teleportPosition = { x = 33937, y = 31074, z = 8 } },
+ { conor = { x = 33937, y = 31098, z = 9 }, teleportPosition = { x = 33929, y = 31109, z = 8 } },
+ { conor = { x = 33933, y = 31109, z = 9 }, teleportPosition = { x = 33929, y = 31109, z = 8 } },
+ { conor = { x = 33921, y = 31113, z = 9 }, teleportPosition = { x = 33929, y = 31109, z = 8 } },
+ { conor = { x = 33912, y = 31113, z = 9 }, teleportPosition = { x = 33904, y = 31117, z = 8 } },
+ { conor = { x = 33901, y = 31108, z = 9 }, teleportPosition = { x = 33904, y = 31117, z = 8 } },
+ { conor = { x = 33901, y = 31098, z = 9 }, teleportPosition = { x = 33904, y = 31082, z = 8 } },
+ { conor = { x = 33899, y = 31064, z = 9 }, teleportPosition = { x = 33904, y = 31082, z = 8 } },
+ },
+ mapsPath = {
+ empty = "data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-empty.otbm",
+ inundate = "data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-inundate.otbm",
+ ebbFlow = "data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow.otbm",
+ },
+
+ -- In Minutes
+ intervalChangeMap = 2,
+ waitPosition = Position(33893, 31020, 8),
+
+ getZone = function()
+ return SoulWarQuest.ebbAndFlow.zone
+ end,
+
+ reloadZone = function()
+ SoulWarQuest.ebbAndFlow.zone:addArea({ x = 33869, y = 30991, z = 8 }, { x = 33964, y = 31147, z = 9 })
+ end,
+
+ kv = KV.scoped("quest"):scoped("soul-war"):scoped("ebb-and-flow-maps"),
+ isActive = function()
+ return SoulWarQuest.ebbAndFlow.kv:get("is-active")
+ end,
+ isLoadedEmptyMap = function()
+ return SoulWarQuest.ebbAndFlow.kv:get("is-loaded-empty-map")
+ end,
+ setActive = function(value)
+ SoulWarQuest.ebbAndFlow.kv:set("is-active", value)
+ end,
+ setLoadedEmptyMap = function(value)
+ SoulWarQuest.ebbAndFlow.kv:set("is-loaded-empty-map", value)
+ end,
+
+ updateZonePlayers = function()
+ if SoulWarQuest.ebbAndFlow.zone and SoulWarQuest.ebbAndFlow.getZone():countPlayers() > 0 then
+ SoulWarQuest.ebbAndFlow.reloadZone()
+ local players = SoulWarQuest.ebbAndFlow.getZone():getPlayers()
+ for _, player in ipairs(players) do
+ logger.trace("Updating player: {}", player:getName())
+ player:sendCreatureAppear()
+ end
+ end
+ end,
+
+ -- Add here more positions of the pools that must transform before innundate map is loaded
+ poolPositions = {
+ { x = 33906, y = 31026, z = 9 },
+ { x = 33901, y = 31026, z = 9 },
+ { x = 33932, y = 31011, z = 9 },
+ { x = 33941, y = 31033, z = 9 },
+ { x = 33946, y = 31037, z = 9 },
+ { x = 33939, y = 31056, z = 9 },
+ },
+
+ boatId = 7272,
+ doorId = 33767,
+ smallPoolId = 33772,
+ MediumPoolId = 33773,
+ },
+
+ changeBlueEvent = nil,
+ changePurpleEvent = nil,
+
+ changeMegalomaniaBlue = function()
+ local boss = Creature("Goshnar's Megalomania")
+ if boss then
+ boss:teleportTo(SoulWarQuest.levers.goshnarsMegalomania.boss.position)
+ boss:say("ENOUGH! I WILL MAKE YOU SUFFER FOR YOUR INSOLENCE! NOW - I - WILL - ANIHILATE - YOU!")
+ boss:setType("Goshnar's Megalomania Blue")
+ local function changeBack()
+ boss:setType("Goshnar's Megalomania Purple")
+ end
+
+ changePurpleEvent = addEvent(changeBack, 7000)
+ end
+ end,
+
+ -- Chance to heal the life of the monster by stepping on the corpse of "weeping soul"
+ goshnarsSpiteHealChance = 10,
+ -- Percentage that will heal by stepping and the chance is successful
+ goshnarsSpiteHealPercentage = 10,
+
+ goshnarSpiteEntrancePosition = { fromPos = Position(33950, 31109, 8), toPos = Position(33780, 31634, 14) },
+
+ waterElementalOutfit = {
+ lookType = 286,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+ },
+
+ goshnarsSpiteFirePositions = {
+ -- North
+ { x = 33743, y = 31628, z = 14 },
+ -- East
+ { x = 33736, y = 31632, z = 14 },
+ -- West
+ { x = 33750, y = 31632, z = 14 },
+ -- South
+ { x = 33742, y = 31637, z = 14 },
+ },
+
+ -- Increased defense if the searing fire disappears
+ goshnarsSpiteIncreaseDefense = 10,
+ -- Count of monsters to kill for enter in the boss room
+ hardozousPanthomDeathCount = 20,
+ -- Time to fire created again
+ timeToCreateSearingFire = 14, -- In seconds
+ -- Time to remove the searing fire if player don't step on it
+ timeToRemoveSearingFire = 5, -- In seconds
+ cooldownToStepOnSearingFire = 56, -- In seconds (14 seconds x 4)
+
+ -- Positions to teleport into rooms when innundate map is loaded
+ ebbAndFlowBoatTeleportPositions = {
+ -- First boat
+ -- Enter on boat
+ { register = { x = 33919, y = 31019, z = 8 }, teleportTo = { x = 33923, y = 31019, z = 8 } },
+ { register = { x = 33919, y = 31020, z = 8 }, teleportTo = { x = 33923, y = 31020, z = 8 } },
+ { register = { x = 33919, y = 31021, z = 8 }, teleportTo = { x = 33923, y = 31021, z = 8 } },
+ { register = { x = 33919, y = 31022, z = 8 }, teleportTo = { x = 33923, y = 31022, z = 8 } },
+ -- Back to innitial room
+ { register = { x = 33922, y = 31019, z = 8 }, teleportTo = { x = 33918, y = 31019, z = 8 } },
+ { register = { x = 33922, y = 31020, z = 8 }, teleportTo = { x = 33918, y = 31020, z = 8 } },
+ { register = { x = 33922, y = 31021, z = 8 }, teleportTo = { x = 33918, y = 31021, z = 8 } },
+ { register = { x = 33922, y = 31022, z = 8 }, teleportTo = { x = 33918, y = 31022, z = 8 } },
+ -- From boat to room
+ { register = { x = 33926, y = 31019, z = 8 }, teleportTo = { x = 33930, y = 31019, z = 8 } },
+ { register = { x = 33926, y = 31020, z = 8 }, teleportTo = { x = 33930, y = 31020, z = 8 } },
+ { register = { x = 33926, y = 31021, z = 8 }, teleportTo = { x = 33930, y = 31021, z = 8 } },
+ { register = { x = 33926, y = 31022, z = 8 }, teleportTo = { x = 33930, y = 31022, z = 8 } },
+ -- From room to boat
+ { register = { x = 33929, y = 31019, z = 8 }, teleportTo = { x = 33925, y = 31019, z = 8 } },
+ { register = { x = 33929, y = 31020, z = 8 }, teleportTo = { x = 33925, y = 31020, z = 8 } },
+ { register = { x = 33929, y = 31021, z = 8 }, teleportTo = { x = 33925, y = 31021, z = 8 } },
+ { register = { x = 33929, y = 31022, z = 8 }, teleportTo = { x = 33925, y = 31022, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33929, y = 31045, z = 8 }, teleportTo = { x = 33925, y = 31045, z = 8 } },
+ { register = { x = 33929, y = 31046, z = 8 }, teleportTo = { x = 33925, y = 31046, z = 8 } },
+ { register = { x = 33929, y = 31047, z = 8 }, teleportTo = { x = 33925, y = 31047, z = 8 } },
+ { register = { x = 33929, y = 31048, z = 8 }, teleportTo = { x = 33925, y = 31048, z = 8 } },
+ -- Back to room
+ { register = { x = 33926, y = 31045, z = 8 }, teleportTo = { x = 33930, y = 31045, z = 8 } },
+ { register = { x = 33926, y = 31046, z = 8 }, teleportTo = { x = 33930, y = 31046, z = 8 } },
+ { register = { x = 33926, y = 31047, z = 8 }, teleportTo = { x = 33930, y = 31047, z = 8 } },
+ { register = { x = 33926, y = 31048, z = 8 }, teleportTo = { x = 33930, y = 31048, z = 8 } },
+ -- From boat to room
+ { register = { x = 33922, y = 31045, z = 8 }, teleportTo = { x = 33918, y = 31045, z = 8 } },
+ { register = { x = 33922, y = 31046, z = 8 }, teleportTo = { x = 33918, y = 31046, z = 8 } },
+ { register = { x = 33922, y = 31047, z = 8 }, teleportTo = { x = 33918, y = 31047, z = 8 } },
+ { register = { x = 33922, y = 31048, z = 8 }, teleportTo = { x = 33918, y = 31048, z = 8 } },
+ -- From room to boat
+ { register = { x = 33919, y = 31045, z = 8 }, teleportTo = { x = 33923, y = 31045, z = 8 } },
+ { register = { x = 33919, y = 31046, z = 8 }, teleportTo = { x = 33923, y = 31046, z = 8 } },
+ { register = { x = 33919, y = 31047, z = 8 }, teleportTo = { x = 33923, y = 31047, z = 8 } },
+ { register = { x = 33919, y = 31048, z = 8 }, teleportTo = { x = 33923, y = 31048, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33896, y = 31055, z = 8 }, teleportTo = { x = 33896, y = 31059, z = 8 } },
+ { register = { x = 33897, y = 31055, z = 8 }, teleportTo = { x = 33897, y = 31059, z = 8 } },
+ { register = { x = 33898, y = 31055, z = 8 }, teleportTo = { x = 33898, y = 31059, z = 8 } },
+ { register = { x = 33899, y = 31055, z = 8 }, teleportTo = { x = 33899, y = 31059, z = 8 } },
+ { register = { x = 33900, y = 31055, z = 8 }, teleportTo = { x = 33900, y = 31059, z = 8 } },
+ { register = { x = 33901, y = 31055, z = 8 }, teleportTo = { x = 33901, y = 31059, z = 8 } },
+ -- Back to room
+ { register = { x = 33896, y = 31058, z = 8 }, teleportTo = { x = 33896, y = 31054, z = 8 } },
+ { register = { x = 33897, y = 31058, z = 8 }, teleportTo = { x = 33897, y = 31054, z = 8 } },
+ { register = { x = 33898, y = 31058, z = 8 }, teleportTo = { x = 33898, y = 31054, z = 8 } },
+ { register = { x = 33899, y = 31058, z = 8 }, teleportTo = { x = 33899, y = 31054, z = 8 } },
+ { register = { x = 33900, y = 31058, z = 8 }, teleportTo = { x = 33900, y = 31054, z = 8 } },
+ { register = { x = 33901, y = 31058, z = 8 }, teleportTo = { x = 33901, y = 31054, z = 8 } },
+ -- From boat to room
+ { register = { x = 33896, y = 31061, z = 8 }, teleportTo = { x = 33896, y = 31065, z = 8 } },
+ { register = { x = 33897, y = 31061, z = 8 }, teleportTo = { x = 33897, y = 31065, z = 8 } },
+ { register = { x = 33898, y = 31061, z = 8 }, teleportTo = { x = 33898, y = 31065, z = 8 } },
+ { register = { x = 33899, y = 31061, z = 8 }, teleportTo = { x = 33899, y = 31065, z = 8 } },
+ { register = { x = 33900, y = 31061, z = 8 }, teleportTo = { x = 33900, y = 31065, z = 8 } },
+ { register = { x = 33901, y = 31061, z = 8 }, teleportTo = { x = 33901, y = 31065, z = 8 } },
+ -- From room to boat
+ { register = { x = 33896, y = 31064, z = 8 }, teleportTo = { x = 33896, y = 31060, z = 8 } },
+ { register = { x = 33897, y = 31064, z = 8 }, teleportTo = { x = 33897, y = 31060, z = 8 } },
+ { register = { x = 33898, y = 31064, z = 8 }, teleportTo = { x = 33898, y = 31060, z = 8 } },
+ { register = { x = 33899, y = 31064, z = 8 }, teleportTo = { x = 33899, y = 31060, z = 8 } },
+ { register = { x = 33900, y = 31064, z = 8 }, teleportTo = { x = 33900, y = 31060, z = 8 } },
+ { register = { x = 33901, y = 31064, z = 8 }, teleportTo = { x = 33901, y = 31060, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33899, y = 31099, z = 8 }, teleportTo = { x = 33899, y = 31103, z = 8 } },
+ { register = { x = 33900, y = 31099, z = 8 }, teleportTo = { x = 33900, y = 31103, z = 8 } },
+ { register = { x = 33901, y = 31099, z = 8 }, teleportTo = { x = 33901, y = 31103, z = 8 } },
+ { register = { x = 33902, y = 31099, z = 8 }, teleportTo = { x = 33902, y = 31103, z = 8 } },
+ { register = { x = 33903, y = 31099, z = 8 }, teleportTo = { x = 33903, y = 31103, z = 8 } },
+ { register = { x = 33904, y = 31099, z = 8 }, teleportTo = { x = 33904, y = 31103, z = 8 } },
+ { register = { x = 33905, y = 31099, z = 8 }, teleportTo = { x = 33905, y = 31103, z = 8 } },
+ -- Back from boat to room
+ { register = { x = 33899, y = 31102, z = 8 }, teleportTo = { x = 33899, y = 31098, z = 8 } },
+ { register = { x = 33900, y = 31102, z = 8 }, teleportTo = { x = 33900, y = 31098, z = 8 } },
+ { register = { x = 33901, y = 31102, z = 8 }, teleportTo = { x = 33901, y = 31098, z = 8 } },
+ { register = { x = 33902, y = 31102, z = 8 }, teleportTo = { x = 33902, y = 31098, z = 8 } },
+ { register = { x = 33903, y = 31102, z = 8 }, teleportTo = { x = 33903, y = 31098, z = 8 } },
+ { register = { x = 33904, y = 31102, z = 8 }, teleportTo = { x = 33904, y = 31098, z = 8 } },
+ { register = { x = 33905, y = 31102, z = 8 }, teleportTo = { x = 33905, y = 31098, z = 8 } },
+ -- From boat to room
+ { register = { x = 33899, y = 31105, z = 8 }, teleportTo = { x = 33899, y = 31109, z = 8 } },
+ { register = { x = 33900, y = 31105, z = 8 }, teleportTo = { x = 33900, y = 31109, z = 8 } },
+ { register = { x = 33901, y = 31105, z = 8 }, teleportTo = { x = 33901, y = 31109, z = 8 } },
+ { register = { x = 33902, y = 31105, z = 8 }, teleportTo = { x = 33902, y = 31109, z = 8 } },
+ { register = { x = 33903, y = 31105, z = 8 }, teleportTo = { x = 33903, y = 31109, z = 8 } },
+ { register = { x = 33904, y = 31105, z = 8 }, teleportTo = { x = 33904, y = 31109, z = 8 } },
+ { register = { x = 33905, y = 31105, z = 8 }, teleportTo = { x = 33905, y = 31109, z = 8 } },
+ -- From room to boat
+ { register = { x = 33899, y = 31108, z = 8 }, teleportTo = { x = 33899, y = 31104, z = 8 } },
+ { register = { x = 33900, y = 31108, z = 8 }, teleportTo = { x = 33900, y = 31104, z = 8 } },
+ { register = { x = 33901, y = 31108, z = 8 }, teleportTo = { x = 33901, y = 31104, z = 8 } },
+ { register = { x = 33902, y = 31108, z = 8 }, teleportTo = { x = 33902, y = 31104, z = 8 } },
+ { register = { x = 33903, y = 31108, z = 8 }, teleportTo = { x = 33903, y = 31104, z = 8 } },
+ { register = { x = 33904, y = 31108, z = 8 }, teleportTo = { x = 33904, y = 31104, z = 8 } },
+ { register = { x = 33905, y = 31108, z = 8 }, teleportTo = { x = 33905, y = 31104, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33913, y = 31112, z = 8 }, teleportTo = { x = 33917, y = 31112, z = 8 } },
+ { register = { x = 33913, y = 31113, z = 8 }, teleportTo = { x = 33917, y = 31113, z = 8 } },
+ { register = { x = 33913, y = 31114, z = 8 }, teleportTo = { x = 33917, y = 31114, z = 8 } },
+ { register = { x = 33913, y = 31115, z = 8 }, teleportTo = { x = 33917, y = 31115, z = 8 } },
+ { register = { x = 33913, y = 31116, z = 8 }, teleportTo = { x = 33917, y = 31116, z = 8 } },
+ -- Back to room
+ { register = { x = 33916, y = 31112, z = 8 }, teleportTo = { x = 33912, y = 31112, z = 8 } },
+ { register = { x = 33916, y = 31113, z = 8 }, teleportTo = { x = 33912, y = 31113, z = 8 } },
+ { register = { x = 33916, y = 31114, z = 8 }, teleportTo = { x = 33912, y = 31114, z = 8 } },
+ { register = { x = 33916, y = 31115, z = 8 }, teleportTo = { x = 33912, y = 31115, z = 8 } },
+ { register = { x = 33916, y = 31116, z = 8 }, teleportTo = { x = 33912, y = 31116, z = 8 } },
+ -- From boat to room
+ { register = { x = 33918, y = 31112, z = 8 }, teleportTo = { x = 33922, y = 31112, z = 8 } },
+ { register = { x = 33918, y = 31113, z = 8 }, teleportTo = { x = 33922, y = 31113, z = 8 } },
+ { register = { x = 33918, y = 31114, z = 8 }, teleportTo = { x = 33922, y = 31114, z = 8 } },
+ { register = { x = 33918, y = 31115, z = 8 }, teleportTo = { x = 33922, y = 31115, z = 8 } },
+ { register = { x = 33918, y = 31116, z = 8 }, teleportTo = { x = 33922, y = 31116, z = 8 } },
+ -- From room to boat
+ { register = { x = 33921, y = 31112, z = 8 }, teleportTo = { x = 33917, y = 31112, z = 8 } },
+ { register = { x = 33921, y = 31113, z = 8 }, teleportTo = { x = 33917, y = 31113, z = 8 } },
+ { register = { x = 33921, y = 31114, z = 8 }, teleportTo = { x = 33917, y = 31114, z = 8 } },
+ { register = { x = 33921, y = 31115, z = 8 }, teleportTo = { x = 33917, y = 31115, z = 8 } },
+ { register = { x = 33921, y = 31116, z = 8 }, teleportTo = { x = 33917, y = 31116, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33936, y = 31087, z = 8 }, teleportTo = { x = 33936, y = 31091, z = 8 } },
+ { register = { x = 33937, y = 31087, z = 8 }, teleportTo = { x = 33937, y = 31091, z = 8 } },
+ { register = { x = 33938, y = 31087, z = 8 }, teleportTo = { x = 33938, y = 31091, z = 8 } },
+ { register = { x = 33939, y = 31087, z = 8 }, teleportTo = { x = 33939, y = 31091, z = 8 } },
+ { register = { x = 33940, y = 31087, z = 8 }, teleportTo = { x = 33940, y = 31091, z = 8 } },
+ { register = { x = 33941, y = 31087, z = 8 }, teleportTo = { x = 33941, y = 31091, z = 8 } },
+ -- Back to room
+ { register = { x = 33936, y = 31090, z = 8 }, teleportTo = { x = 33936, y = 31086, z = 8 } },
+ { register = { x = 33937, y = 31090, z = 8 }, teleportTo = { x = 33937, y = 31086, z = 8 } },
+ { register = { x = 33938, y = 31090, z = 8 }, teleportTo = { x = 33938, y = 31086, z = 8 } },
+ { register = { x = 33939, y = 31090, z = 8 }, teleportTo = { x = 33939, y = 31086, z = 8 } },
+ { register = { x = 33940, y = 31090, z = 8 }, teleportTo = { x = 33940, y = 31086, z = 8 } },
+ { register = { x = 33941, y = 31090, z = 8 }, teleportTo = { x = 33941, y = 31086, z = 8 } },
+ -- From boat to room
+ { register = { x = 33936, y = 31095, z = 8 }, teleportTo = { x = 33934, y = 31099, z = 8 } },
+ { register = { x = 33937, y = 31095, z = 8 }, teleportTo = { x = 33935, y = 31099, z = 8 } },
+ { register = { x = 33938, y = 31095, z = 8 }, teleportTo = { x = 33936, y = 31099, z = 8 } },
+ { register = { x = 33939, y = 31095, z = 8 }, teleportTo = { x = 33937, y = 31099, z = 8 } },
+ { register = { x = 33940, y = 31095, z = 8 }, teleportTo = { x = 33938, y = 31099, z = 8 } },
+ { register = { x = 33941, y = 31095, z = 8 }, teleportTo = { x = 33939, y = 31099, z = 8 } },
+ -- From room to boat
+ { register = { x = 33934, y = 31098, z = 8 }, teleportTo = { x = 33936, y = 31094, z = 8 } },
+ { register = { x = 33935, y = 31098, z = 8 }, teleportTo = { x = 33937, y = 31094, z = 8 } },
+ { register = { x = 33936, y = 31098, z = 8 }, teleportTo = { x = 33938, y = 31094, z = 8 } },
+ { register = { x = 33937, y = 31098, z = 8 }, teleportTo = { x = 33939, y = 31094, z = 8 } },
+ { register = { x = 33938, y = 31098, z = 8 }, teleportTo = { x = 33940, y = 31094, z = 8 } },
+ { register = { x = 33939, y = 31098, z = 8 }, teleportTo = { x = 33941, y = 31094, z = 8 } },
+ { register = { x = 33940, y = 31098, z = 8 }, teleportTo = { x = 33942, y = 31094, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33939, y = 31064, z = 8 }, teleportTo = { x = 33939, y = 31060, z = 8 } },
+ { register = { x = 33940, y = 31064, z = 8 }, teleportTo = { x = 33940, y = 31060, z = 8 } },
+ { register = { x = 33941, y = 31064, z = 8 }, teleportTo = { x = 33941, y = 31060, z = 8 } },
+ { register = { x = 33942, y = 31064, z = 8 }, teleportTo = { x = 33942, y = 31060, z = 8 } },
+ { register = { x = 33943, y = 31064, z = 8 }, teleportTo = { x = 33943, y = 31060, z = 8 } },
+ { register = { x = 33944, y = 31064, z = 8 }, teleportTo = { x = 33944, y = 31060, z = 8 } },
+ -- Back to room
+ { register = { x = 33939, y = 31061, z = 8 }, teleportTo = { x = 33939, y = 31065, z = 8 } },
+ { register = { x = 33940, y = 31061, z = 8 }, teleportTo = { x = 33940, y = 31065, z = 8 } },
+ { register = { x = 33941, y = 31061, z = 8 }, teleportTo = { x = 33941, y = 31065, z = 8 } },
+ { register = { x = 33942, y = 31061, z = 8 }, teleportTo = { x = 33942, y = 31065, z = 8 } },
+ { register = { x = 33943, y = 31061, z = 8 }, teleportTo = { x = 33943, y = 31065, z = 8 } },
+ { register = { x = 33944, y = 31061, z = 8 }, teleportTo = { x = 33944, y = 31065, z = 8 } },
+ -- From boat to room
+ { register = { x = 33939, y = 31058, z = 8 }, teleportTo = { x = 33939, y = 31054, z = 8 } },
+ { register = { x = 33940, y = 31058, z = 8 }, teleportTo = { x = 33940, y = 31054, z = 8 } },
+ { register = { x = 33941, y = 31058, z = 8 }, teleportTo = { x = 33941, y = 31054, z = 8 } },
+ { register = { x = 33942, y = 31058, z = 8 }, teleportTo = { x = 33942, y = 31054, z = 8 } },
+ { register = { x = 33943, y = 31058, z = 8 }, teleportTo = { x = 33943, y = 31054, z = 8 } },
+ { register = { x = 33944, y = 31058, z = 8 }, teleportTo = { x = 33944, y = 31054, z = 8 } },
+ -- From room to boat
+ { register = { x = 33939, y = 31055, z = 8 }, teleportTo = { x = 33939, y = 31059, z = 8 } },
+ { register = { x = 33940, y = 31055, z = 8 }, teleportTo = { x = 33940, y = 31059, z = 8 } },
+ { register = { x = 33941, y = 31055, z = 8 }, teleportTo = { x = 33941, y = 31059, z = 8 } },
+ { register = { x = 33942, y = 31055, z = 8 }, teleportTo = { x = 33942, y = 31059, z = 8 } },
+ { register = { x = 33943, y = 31055, z = 8 }, teleportTo = { x = 33943, y = 31059, z = 8 } },
+ { register = { x = 33944, y = 31055, z = 8 }, teleportTo = { x = 33944, y = 31059, z = 8 } },
+
+ -- Boat
+ -- Enter on boat
+ { register = { x = 33934, y = 31108, z = 8 }, teleportTo = { x = 33938, y = 31108, z = 8 } },
+ { register = { x = 33934, y = 31109, z = 8 }, teleportTo = { x = 33938, y = 31109, z = 8 } },
+ { register = { x = 33934, y = 31110, z = 8 }, teleportTo = { x = 33938, y = 31110, z = 8 } },
+ { register = { x = 33934, y = 31111, z = 8 }, teleportTo = { x = 33938, y = 31111, z = 8 } },
+ { register = { x = 33934, y = 31112, z = 8 }, teleportTo = { x = 33938, y = 31112, z = 8 } },
+ -- Back to room
+ { register = { x = 33937, y = 31108, z = 8 }, teleportTo = { x = 33933, y = 31108, z = 8 } },
+ { register = { x = 33937, y = 31109, z = 8 }, teleportTo = { x = 33933, y = 31109, z = 8 } },
+ { register = { x = 33937, y = 31110, z = 8 }, teleportTo = { x = 33933, y = 31110, z = 8 } },
+ { register = { x = 33937, y = 31111, z = 8 }, teleportTo = { x = 33933, y = 31111, z = 8 } },
+ { register = { x = 33937, y = 31112, z = 8 }, teleportTo = { x = 33933, y = 31112, z = 8 } },
+ -- From boat to room
+ { register = { x = 33942, y = 31108, z = 8 }, teleportTo = { x = 33946, y = 31108, z = 8 } },
+ { register = { x = 33942, y = 31109, z = 8 }, teleportTo = { x = 33946, y = 31109, z = 8 } },
+ { register = { x = 33942, y = 31110, z = 8 }, teleportTo = { x = 33946, y = 31110, z = 8 } },
+ { register = { x = 33942, y = 31111, z = 8 }, teleportTo = { x = 33946, y = 31111, z = 8 } },
+ { register = { x = 33942, y = 31112, z = 8 }, teleportTo = { x = 33946, y = 31112, z = 8 } },
+ -- From room to boat
+ { register = { x = 33945, y = 31108, z = 8 }, teleportTo = { x = 33941, y = 31108, z = 8 } },
+ { register = { x = 33945, y = 31109, z = 8 }, teleportTo = { x = 33941, y = 31109, z = 8 } },
+ { register = { x = 33945, y = 31110, z = 8 }, teleportTo = { x = 33941, y = 31110, z = 8 } },
+ { register = { x = 33945, y = 31111, z = 8 }, teleportTo = { x = 33941, y = 31111, z = 8 } },
+ { register = { x = 33945, y = 31112, z = 8 }, teleportTo = { x = 33941, y = 31112, z = 8 } },
+ },
+}
+
+function RegisterSoulWarBossesLevers()
+ -- Register levers
+ local goshnarsMaliceLever = BossLever(SoulWarQuest.levers.goshnarsMalice)
+ goshnarsMaliceLever:position(SoulWarQuest.levers.goshnarsMalicePosition)
+ goshnarsMaliceLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsMaliceLever:getZone():getName())
+
+ local goshnarsSpiteLever = BossLever(SoulWarQuest.levers.goshnarsSpite)
+ goshnarsSpiteLever:position(SoulWarQuest.levers.goshnarsSpitePosition)
+ goshnarsSpiteLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsSpiteLever:getZone():getName())
+
+ local goshnarsGreedLever = BossLever(SoulWarQuest.levers.goshnarsGreed)
+ goshnarsGreedLever:position(SoulWarQuest.levers.goshnarsGreedPosition)
+ goshnarsGreedLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsGreedLever:getZone():getName())
+
+ local goshnarsHatredLever = BossLever(SoulWarQuest.levers.goshnarsHatred)
+ goshnarsHatredLever:position(SoulWarQuest.levers.goshnarsHatredPosition)
+ goshnarsHatredLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsHatredLever:getZone():getName())
+
+ local goshnarsCrueltyLever = BossLever(SoulWarQuest.levers.goshnarsCruelty)
+ goshnarsCrueltyLever:position(SoulWarQuest.levers.goshnarsCrueltyPosition)
+ goshnarsCrueltyLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsCrueltyLever:getZone():getName())
+
+ local goshnarsMegalomaniaLever = BossLever(SoulWarQuest.levers.goshnarsMegalomania)
+ goshnarsMegalomaniaLever:position(SoulWarQuest.levers.goshnarsMegalomaniaPosition)
+ goshnarsMegalomaniaLever:register()
+ logger.debug("Registering soul war boss lever zone: {}", goshnarsMegalomaniaLever:getZone():getName())
+end
+
+-- Initialize ebb and flow zone area
+SoulWarQuest.ebbAndFlow.zone:addArea({ x = 33869, y = 30991, z = 8 }, { x = 33964, y = 31147, z = 9 })
+
+-- Initialize claustrophobic inferno raid zones and add remove destination
+
+for _, raid in ipairs(SoulWarQuest.claustrophobicInfernoRaids) do
+ local zone = raid.getZone()
+ zone:addArea(raid.zoneArea[1], raid.zoneArea[2])
+ zone:setRemoveDestination(raid.exitPosition)
+end
+
+-- Initialize bosses access for taint check
+SoulWarQuest.areaZones.claustrophobicInferno:addArea({ x = 33982, y = 30981, z = 9 }, { x = 34051, y = 31110, z = 11 })
+
+SoulWarQuest.areaZones.ebbAndFlow:addArea({ x = 33873, y = 30994, z = 8 }, { x = 33968, y = 31150, z = 9 })
+
+SoulWarQuest.areaZones.furiousCrater:addArea({ x = 33814, y = 31819, z = 3 }, { x = 33907, y = 31920, z = 7 })
+
+SoulWarQuest.areaZones.rottenWasteland:addArea({ x = 33980, y = 30986, z = 11 }, { x = 33901, y = 31105, z = 12 })
+
+SoulWarQuest.areaZones.mirroredNightmare:addArea({ x = 33877, y = 31164, z = 9 }, { x = 33991, y = 31241, z = 13 })
+
+-- Initialize safe areas (should not spawn monster, teleport, take damage from taint, etc)
+SoulWarQuest.areaZones.claustrophobicInferno:subtractArea({ x = 34002, y = 31008, z = 9 }, { x = 34019, y = 31019, z = 9 })
+
+SoulWarQuest.areaZones.ebbAndFlow:subtractArea({ x = 33887, y = 31015, z = 8 }, { x = 33920, y = 31024, z = 8 })
+
+SoulWarQuest.areaZones.furiousCrater:subtractArea({ x = 33854, y = 31828, z = 3 }, { x = 33869, y = 31834, z = 3 })
+
+SoulWarQuest.areaZones.rottenWasteland:subtractArea({ x = 33967, y = 31037, z = 11 }, { x = 33977, y = 31051, z = 11 })
+
+SoulWarQuest.areaZones.mirroredNightmare:subtractArea({ x = 33884, y = 31181, z = 10 }, { x = 33892, y = 31198, z = 10 })
+
+SoulCagePosition = Position(33709, 31596, 14)
+TaintDurationSeconds = 14 * 24 * 60 * 60 -- 14 days
+GreedbeastKills = 0
+
+SoulWarReflectDamageMap = {
+ [COMBAT_PHYSICALDAMAGE] = 10,
+ [COMBAT_FIREDAMAGE] = 10,
+ [COMBAT_EARTHDAMAGE] = 10,
+ [COMBAT_ENERGYDAMAGE] = 10,
+ [COMBAT_ICEDAMAGE] = 10,
+ [COMBAT_HOLYDAMAGE] = 10,
+ [COMBAT_DEATHDAMAGE] = 10,
+}
+
+local soulWarTaints = {
+ "taints-teleport", -- Taint 1
+ "taints-spawn", -- Taint 2
+ "taints-damage", -- Taint 3
+ "taints-heal", -- Taint 4
+ "taints-loss", -- Taint 5
+}
+
+GreedMonsters = {
+ ["Greedbeast"] = Position(33744, 31666, 14),
+ ["Soulsnatcher"] = Position(33747, 31668, 14),
+ ["Weak Soul"] = Position(33750, 31666, 14),
+ ["Strong Soul"] = Position(33750, 31666, 14),
+ ["Powerful Soul"] = Position(33750, 31666, 14),
+}
+
+function CreateGoshnarsGreedMonster(name, position)
+ local function sendEffect()
+ position:sendMagicEffect(CONST_ME_TELEPORT)
+ end
+
+ local function spawnMonster()
+ Game.createMonster(name, position, true, false)
+ logger.trace("Spawning {} in position {}", name, position:toString())
+ end
+
+ for i = 7, 9 do
+ addEvent(sendEffect, i * 1000)
+ end
+
+ addEvent(spawnMonster, 10000)
+end
+
+function RemoveSoulCageAndBuffMalice()
+ local soulCage = Creature("Soul Cage")
+ if soulCage then
+ soulCage:remove()
+ addEvent(SpawnSoulCage, 23000)
+ local malice = Creature("Goshnar's Malice")
+ if malice then
+ logger.trace("Found malice, try adding reflect and defense")
+ for elementType, reflectPercent in pairs(SoulWarReflectDamageMap) do
+ malice:addReflectElement(elementType, reflectPercent)
+ end
+ malice:addDefense(10)
+ end
+ end
+end
+
+function SpawnSoulCage()
+ local tile = Tile(SoulCagePosition)
+ local creatures = tile:getCreatures() or {}
+ local soulCage = Creature("Soul Cage")
+ if not soulCage then
+ Game.createMonster("Soul Cage", SoulCagePosition, true, true)
+ logger.trace("Spawning Soul Cage in position {}", SoulCagePosition:toString())
+ addEvent(RemoveSoulCageAndBuffMalice, 40000)
+ end
+end
+
+local function shuffle(list)
+ for i = #list, 2, -1 do
+ local j = math.random(i)
+ list[i], list[j] = list[j], list[i]
+ end
+end
+
+local function createConnectedGroup(startPos, groupPositions, groupSize)
+ local group = { startPos }
+ local lastPos = startPos
+ local directions = {
+ { x = 1, y = 0 },
+ { x = -1, y = 0 }, -- Right and left
+ { x = 0, y = 1 },
+ { x = 0, y = -1 }, -- Up and down
+ { x = 1, y = 1 },
+ { x = -1, y = -1 }, -- Diagonals
+ { x = -1, y = 1 },
+ { x = 1, y = -1 },
+ }
+
+ for i = 2, groupSize do
+ shuffle(directions)
+ local nextPos = nil
+ for _, dir in ipairs(directions) do
+ local potentialNextPos = Position(lastPos.x + dir.x, lastPos.y + dir.y, lastPos.z)
+ if table.contains(groupPositions, potentialNextPos) then
+ nextPos = potentialNextPos
+ break
+ end
+ end
+
+ if nextPos then
+ table.insert(group, nextPos)
+ table.remove(groupPositions, table.find(groupPositions, nextPos))
+ lastPos = nextPos
+ else
+ break
+ end
+ end
+
+ return group
+end
+
+local function generatePositionsInRange(center, range)
+ local positions = {}
+ for x = center.x - range, center.x + range do
+ for y = center.y - range, center.y + range do
+ table.insert(positions, Position(x, y, center.z))
+ end
+ end
+ return positions
+end
+
+local toRevertPositions = {}
+
+local tileItemIds = {
+ 32906,
+ 33066,
+ 33067,
+ 33068,
+ 33069,
+ 33070,
+}
+
+local function revertTilesAndApplyDamage(zonePositions)
+ for _, pos in ipairs(zonePositions) do
+ local tile = Tile(pos)
+ if tile and tile:getGround() then
+ if tile:getGround():getId() ~= 409 then
+ local creature = tile:getTopCreature()
+ if creature then
+ local player = creature:getPlayer()
+ if player then
+ player:addHealth(-8000, COMBAT_DEATHDAMAGE)
+ end
+ end
+ end
+
+ local itemFound = false
+ for i = 1, #tileItemIds do
+ local item = tile:getItemById(tileItemIds[i])
+ if item then
+ itemFound = true
+ break
+ end
+ end
+
+ if tile:getGround():getId() == 410 and not itemFound and not tile:getItemByTopOrder(1) and not tile:getItemByTopOrder(3) then
+ pos:sendMagicEffect(CONST_ME_REDSMOKE)
+ end
+ end
+ end
+
+ for posString, itemId in pairs(toRevertPositions) do
+ local pos = posString:toPosition()
+ local tile = Tile(pos)
+ if tile and tile:getGround() and tile:getGround():getId() == 409 then
+ tile:getGround():transform(itemId)
+ toRevertPositions[pos:toString()] = nil
+ end
+ end
+end
+
+function Monster:createSoulWarWhiteTiles(centerRoomPosition, zonePositions, executeInterval)
+ local groupPositions = generatePositionsInRange(centerRoomPosition, 7)
+ local totalTiles = 11
+ local groupSize = 3
+ local groupsCreated = 0
+
+ -- Run only for megalomania boss
+ if executeInterval then
+ -- Remove remains
+ for _, pos in ipairs(zonePositions) do
+ local tile = Tile(pos)
+ if tile and tile:getGround() then
+ local remains = tile:getItemById(33984)
+ if remains then
+ remains:remove()
+ end
+ end
+ end
+ end
+
+ while #groupPositions > 0 and groupsCreated * groupSize < totalTiles do
+ local randomIndex = math.random(#groupPositions)
+ local startPos = groupPositions[randomIndex]
+ table.remove(groupPositions, randomIndex)
+
+ local group = createConnectedGroup(startPos, groupPositions, groupSize)
+ for _, pos in ipairs(group) do
+ local tile = Tile(pos)
+ if tile then
+ toRevertPositions[pos:toString()] = tile:getGround():getId()
+ tile:getGround():transform(409)
+ end
+ end
+
+ groupsCreated = groupsCreated + 1
+ end
+
+ addEvent(revertTilesAndApplyDamage, executeInterval or 3000, zonePositions)
+end
+
+function MonsterType:calculateBagYouDesireChance(player, itemChance)
+ local playerTaintLevel = player:getTaintLevel()
+ if not playerTaintLevel or playerTaintLevel == 0 then
+ return itemChance
+ end
+
+ local monsterName = self:getName()
+ local isMonsterValid = table.contains(SoulWarQuest.bagYouDesireMonsters, monsterName)
+ if not isMonsterValid then
+ return itemChance
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ local megalomaniaKills = soulWarQuest:scoped("megalomania-kills"):get("count") or 0
+
+ if monsterName == "Goshnar's Megalomania" then
+ -- Special handling for Goshnar's Megalomania
+ itemChance = itemChance + megalomaniaKills * SoulWarQuest.bagYouDesireChancePerTaint
+ else
+ -- General handling for other monsters (bosses and non-bosses)
+ itemChance = itemChance + (playerTaintLevel * SoulWarQuest.bagYouDesireChancePerTaint)
+ end
+
+ logger.info("Player {} killed {} with {} taints, loot chance {}", player:getName(), monsterName, playerTaintLevel, itemChance)
+
+ if math.random(1, 100000) <= totalChance then
+ logger.debug("Player {} killed {} and got a bag you desire with drop chance {}", player:getName(), monsterName, itemChance)
+ if monsterName == "Goshnar's Megalomania" then
+ -- Reset kill count on successful drop
+ soulWarQuest:scoped("megalomania-kills"):set("count", 0)
+ end
+ else
+ if monsterName == "Goshnar's Megalomania" then
+ -- Increment kill count for unsuccessful attempts
+ soulWarQuest:scoped("megalomania-kills"):set("count", megalomaniaKills + 1)
+ end
+ end
+
+ return itemChance
+end
+
+local intervalBetweenExecutions = 10000
+
+local accumulatedTime = 0
+local desiredInterval = 40000
+local bossSayInterval = 38000
+
+function Monster:onThinkMegalomaniaWhiteTiles(interval, zonePositions, revertTime)
+ self:onThinkGoshnarTormentCounter(interval, 36, intervalBetweenExecutions, SoulWarQuest.levers.goshnarsMegalomania.boss.position)
+
+ accumulatedTime = accumulatedTime + interval
+
+ if accumulatedTime == bossSayInterval then
+ self:say("FEEL THE POWER OF MY WRATH!!")
+ end
+ -- Execute only after 40 seconds
+ if accumulatedTime >= desiredInterval then
+ self:createSoulWarWhiteTiles(SoulWarQuest.levers.goshnarsMegalomania.boss.position, zonePositions, revertTime)
+ accumulatedTime = 0
+ end
+end
+
+TaintTeleportCooldown = {}
+
+function Player:getTaintNameByNumber(taintNumber, skipKvCheck)
+ local haveTaintName = nil
+ local soulWarQuest = self:soulWarQuestKV()
+ local taintName = soulWarTaints[taintNumber]
+ if skipKvCheck or taintName and soulWarQuest:get(taintName) then
+ haveTaintName = taintName
+ end
+
+ return haveTaintName
+end
+
+function Player:addNextTaint()
+ local soulWarQuest = self:soulWarQuestKV()
+ for _, taintName in ipairs(soulWarTaints) do
+ if not soulWarQuest:get(taintName) then
+ soulWarQuest:set(taintName, true)
+ self:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have gained the " .. taintName .. ".")
+ self:setTaintIcon()
+ break
+ end
+ end
+end
+
+function Player:setTaintIcon(taintId)
+ self:resetTaintConditions()
+ local condition = Condition(CONDITION_GOSHNARTAINT, CONDITIONID_DEFAULT, taintId or self:getTaintLevel())
+ condition:setTicks(14 * 24 * 60 * 60 * 1000)
+ self:addCondition(condition)
+end
+
+function Player:resetTaintConditions()
+ for i = 1, 5 do
+ self:removeCondition(CONDITION_GOSHNARTAINT, CONDITIONID_DEFAULT, i)
+ end
+end
+
+function Player:getTaintLevel()
+ local taintLevel = nil
+ local soulWarQuest = self:soulWarQuestKV()
+ for i, taint in ipairs(soulWarTaints) do
+ if soulWarQuest:get(taint) then
+ taintLevel = i
+ end
+ end
+
+ return taintLevel
+end
+
+function Player:resetTaints(skipCheckTime)
+ local soulWarQuest = self:soulWarQuestKV()
+ local firstTaintTime = soulWarQuest:get("firstTaintTime")
+ if skipCheckTime or firstTaintTime and os.time() >= (firstTaintTime + TaintDurationSeconds) then
+ -- Reset all taints and remove condition
+ for _, taintName in ipairs(soulWarTaints) do
+ if soulWarQuest:get(taintName) then
+ soulWarQuest:remove(taintName)
+ end
+ end
+ self:resetTaintConditions()
+ soulWarQuest:remove("firstTaintTime")
+ local resetMessage = "Your Goshnar's taints have been reset."
+ if not skipCheckTime then
+ resetMessage = resetMessage .. " You didn't finish the quest in 14 days."
+ end
+ self:sendTextMessage(MESSAGE_EVENT_ADVANCE, resetMessage)
+
+ for bossName, _ in pairs(SoulWarQuest.miniBosses) do
+ soulWarQuest:remove(bossName)
+ end
+ end
+end
+
+function Monster:tryTeleportToPlayer(sayMessage)
+ local range = 30
+ local spectators = Game.getSpectators(self:getPosition(), false, false, range, range, range, range)
+ local maxDistance = 0
+ local farthestPlayer = nil
+ for i, spectator in ipairs(spectators) do
+ if spectator:isPlayer() then
+ local player = spectator:getPlayer()
+ if player:getTaintNameByNumber(1, true) and player:getSoulWarZoneMonster() ~= nil then
+ local distance = self:getPosition():getDistance(player:getPosition())
+ if distance > maxDistance then
+ maxDistance = distance
+ farthestPlayer = player
+ logger.trace("Found player {} to teleport", player:getName())
+ end
+ end
+ end
+ end
+
+ if farthestPlayer and math.random(100) <= 10 then
+ local playerPosition = farthestPlayer:getPosition()
+ if TaintTeleportCooldown[farthestPlayer:getId()] then
+ logger.trace("Cooldown is active to player {}", farthestPlayer:getName())
+ return
+ end
+
+ if not TaintTeleportCooldown[farthestPlayer:getId()] then
+ TaintTeleportCooldown[farthestPlayer:getId()] = true
+
+ logger.trace("Scheduling player {} to teleport", farthestPlayer:getName())
+ self:getPosition():sendMagicEffect(CONST_ME_MORTAREA)
+ farthestPlayer:getPosition():sendMagicEffect(CONST_ME_MORTAREA)
+ addEvent(function(playerId, monsterId)
+ local monsterEvent = Monster(monsterId)
+ local playerEvent = Player(playerId)
+ if monsterEvent and playerEvent then
+ local destinationTile = Tile(playerPosition)
+ if destinationTile and not (destinationTile:hasProperty(CONST_PROP_BLOCKPROJECTILE) or destinationTile:hasProperty(CONST_PROP_MOVEABLE)) then
+ monsterEvent:say(sayMessage)
+ monsterEvent:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ monsterEvent:teleportTo(playerPosition, true)
+ monsterEvent:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ end
+ end
+ end, 2000, farthestPlayer:getId(), self:getId())
+
+ addEvent(function(playerId)
+ local playerEvent = Player(playerId)
+ if not playerEvent then
+ return
+ end
+
+ logger.trace("Cleaning player cooldown")
+ TaintTeleportCooldown[playerEvent:getId()] = nil
+ end, 10000, farthestPlayer:getId())
+ end
+ end
+end
+
+function Monster:getSoulWarKV()
+ return SoulWarQuest.kvSoulWar:scoped("monster"):scoped(self:getName())
+end
+
+function Monster:getHatredDamageMultiplier()
+ return self:getSoulWarKV():get("burning-hatred-empowered") or 0
+end
+
+function Monster:increaseHatredDamageMultiplier(multiplierCount)
+ local attackMultiplier = self:getHatredDamageMultiplier()
+ self:getSoulWarKV():set("burning-hatred-empowered", attackMultiplier + multiplierCount)
+end
+
+function Monster:resetHatredDamageMultiplier()
+ self:getSoulWarKV():remove("burning-hatred-empowered")
+end
+
+function Position:increaseNecromaticMegalomaniaStrength()
+ local tile = Tile(self)
+ if tile then
+ local item = tile:getItemById(SoulWarQuest.necromanticRemainsId)
+ if item then
+ local boss = Creature("Goshnar's Megalomania")
+ if boss then
+ boss:increaseHatredDamageMultiplier(5)
+ item:remove()
+ logger.trace("Necromantic remains strength increased")
+ end
+ end
+ end
+end
+
+local lastExecutionTime = 0
+
+-- Damage 24 to 36 have a special damage
+local damageTable = {
+ 1400,
+ 1600,
+ 1800,
+ 2200,
+ 2400,
+ 2600,
+ 3000,
+ 3400,
+ 3800,
+ 4200,
+ 4800,
+ 5200,
+ 5600,
+}
+
+function Monster:onThinkGoshnarTormentCounter(interval, maxLimit, intervalBetweenExecutions, bossPosition)
+ local interval = os.time() * 1000
+ if interval - lastExecutionTime < intervalBetweenExecutions then
+ return
+ end
+
+ lastExecutionTime = interval
+ logger.trace("Icon time count {}", interval)
+ local spectators = Game.getSpectators(bossPosition, false, true, 15, 15, 15, 15)
+ for i = 1, #spectators do
+ local player = spectators[i]
+ local tormentCounter = player:getGoshnarSymbolTormentCounter()
+ local goshnarsHatred = Creature(bossName or "Goshnar's Megalomania")
+ if not goshnarsHatred then
+ player:resetGoshnarSymbolTormentCounter()
+ goto continue
+ end
+
+ if tormentCounter <= maxLimit then
+ player:increaseGoshnarSymbolTormentCounter(maxLimit)
+ logger.trace("Player {} has {} damage counter", player:getName(), tormentCounter)
+
+ if tormentCounter > 0 then
+ local damage = tormentCounter * 35
+ if tormentCounter >= 24 then
+ damage = damageTable[tormentCounter - 23]
+ end
+
+ logger.trace("Final damage {}", damage)
+ player:addHealth(-damage, COMBAT_DEATHDAMAGE)
+ player:getPosition():sendMagicEffect(CONST_ME_PINK_ENERGY_SPARK)
+ end
+ end
+
+ if tormentCounter == 5 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The dread starts to torment you! Don't let dread level reach critical value!")
+ elseif tormentCounter == 15 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The dread's torment becomes unbearable!")
+ elseif tormentCounter == 24 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The Dread's torment begins to tear you apart!")
+ elseif tormentCounter == 30 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The dread's torment is killing you!")
+ elseif tormentCounter == 36 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The dread's torment is now lethal!")
+ end
+
+ ::continue::
+ end
+end
+
+function Monster:increaseAspectOfPowerDeathCount()
+ local bossKV = self:getSoulWarKV()
+ local aspectDeathCount = bossKV:get("aspect-of-power-death-count") or 0
+ local newCount = aspectDeathCount + 1
+ logger.trace("Aspect of Power death count {}", newCount)
+ bossKV:set("aspect-of-power-death-count", newCount)
+ if newCount == 4 then
+ self:setType("Goshnar's Megalomania Green")
+ self:say("THE DEATH OF ASPECTS DIMINISHES GOSHNAR'S POWER AND HE TURNS VULNERABLE!")
+ bossKV:set("aspect-of-power-death-count", 0)
+ SoulWarQuest.changeBlueEvent = addEvent(SoulWarQuest.changeMegalomaniaBlue, 1 * 60 * 1000)
+ logger.trace("Aspect of Power defeated all and Megalomania is now vulnerable, reseting death count.")
+ SoulWarQuest.changePurpleEvent = addEvent(function()
+ local boss = Creature("Goshnar's Megalomania")
+ if boss and boss:getTypeName() == "Goshnar's Megalomania Green" then
+ boss:setType("Goshnar's Megalomania Purple")
+ boss:say("GOSHNAR REGAINED ENOUGH POWER TO TURN INVULNERABLE AGAIN!")
+ logger.trace("Megalomania is now immune again")
+ end
+ end, SoulWarQuest.timeToReturnImmuneMegalomania * 1000)
+ end
+end
+
+function Monster:goshnarsDefenseIncrease(kvName)
+ local currentTime = os.time()
+ -- Gets the time when the "Greedy Maw" item was last used.
+ local lastItemUseTime = SoulWarQuest.kvSoulWar:get(kvName) or 0
+ -- Checks if more than config time have passed since the item was last used.
+ if currentTime >= lastItemUseTime + SoulWarQuest.timeToIncreaseCrueltyDefense then
+ self:addDefense(SoulWarQuest.goshnarsCrueltyDefenseChange)
+ -- Register the drain callback to modify the damage for goshnar's cruelty
+ local newValue = SoulWarQuest.kvSoulWar:get("goshnars-cruelty-defense-drain") or SoulWarQuest.goshnarsCrueltyDefenseChange
+ SoulWarQuest.kvSoulWar:set("goshnars-cruelty-defense-drain", newValue + 1) -- Increment the value to track usage or modifications
+
+ --- Updates the KV to reflect the timing of the increase to maintain control.
+ SoulWarQuest.kvSoulWar:set(kvName, currentTime)
+ else
+ -- If config time have not passed, logs the increase has been skipped.
+ logger.trace("{} skips increase cooldown due to recent item use.", self:getName())
+ end
+end
+
+function Monster:removeGoshnarsMegalomaniaMonsters(zone)
+ if self:getName() ~= "Goshnar's Megalomania" then
+ return
+ end
+
+ if zone then
+ local creatures = zone:getCreatures()
+ for _, creature in ipairs(creatures) do
+ if creature:getMonster() then
+ creature:remove()
+ end
+ end
+ end
+end
+
+function Player:getSoulWarZoneMonster()
+ local zoneMonsterName = nil
+ for zoneName, monsterName in pairs(SoulWarQuest.areaZones.monsters) do
+ local zone = Zone.getByName(zoneName)
+ if zone and zone:isInZone(self:getPosition()) then
+ zoneMonsterName = monsterName
+ break
+ end
+ end
+
+ return zoneMonsterName
+end
+
+function Player:isInBoatSpot()
+ -- Get ebb and flow zone and check if player is in zone
+ local zone = SoulWarQuest.ebbAndFlow.getZone()
+ local tile = Tile(self:getPosition())
+ local groundId
+ if tile and tile:getGround() then
+ groundId = tile:getGround():getId()
+ end
+ if zone and zone:isInZone(self:getPosition()) and tile and groundId == SoulWarQuest.ebbAndFlow.boatId then
+ logger.trace("Player {} is in boat spot", self:getName())
+ return true
+ end
+
+ logger.trace("Player {} is not in boat spot", self:getName())
+ return false
+end
+
+function Player:soulWarQuestKV()
+ return self:kv():scoped("quest"):scoped("soul-war")
+end
+
+function Player:getGoshnarSymbolTormentCounter()
+ local soulWarKV = self:soulWarQuestKV()
+ return soulWarKV:get("goshnars-hatred-torment-count") or 0
+end
+
+function Player:increaseGoshnarSymbolTormentCounter(maxLimit)
+ local soulWarKV = self:soulWarQuestKV()
+ local tormentCount = self:getGoshnarSymbolTormentCounter()
+ if tormentCount == maxLimit then
+ self:setIcon("goshnars-hatred-damage", CreatureIconCategory_Quests, CreatureIconQuests_RedCross, tormentCount)
+ return
+ end
+
+ self:setIcon("goshnars-hatred-damage", CreatureIconCategory_Quests, CreatureIconQuests_RedCross, tormentCount + 1)
+ soulWarKV:set("goshnars-hatred-torment-count", tormentCount + 1)
+end
+
+function Player:removeGoshnarSymbolTormentCounter(count)
+ local soulWarKV = self:soulWarQuestKV()
+ local tormentCount = self:getGoshnarSymbolTormentCounter()
+ if tormentCount > count then
+ self:setIcon("goshnars-hatred-damage", CreatureIconCategory_Quests, CreatureIconQuests_RedCross, tormentCount - count)
+ soulWarKV:set("goshnars-hatred-torment-count", tormentCount - count)
+ else
+ self:resetGoshnarSymbolTormentCounter()
+ end
+end
+
+function Player:resetGoshnarSymbolTormentCounter()
+ local soulWarKV = self:soulWarQuestKV()
+ soulWarKV:remove("goshnars-hatred-torment-count")
+ self:removeIcon("goshnars-hatred-damage")
+end
+
+function Player:furiousCraterKV()
+ return self:soulWarQuestKV():scoped("furius-crater")
+end
+
+function Player:pulsatingEnergyKV()
+ return self:furiousCraterKV():scoped("pulsating-energy")
+end
+
+function Zone:getRandomPlayer()
+ local players = self:getPlayers()
+ if #players == 0 then
+ return nil
+ end
+
+ local randomIndex = math.random(#players)
+ return players[randomIndex]
+end
+
+local conditionOutfit = Condition(CONDITION_OUTFIT)
+
+local function delayedCastSpell(cid, var, combat, targetId)
+ local creature = Creature(cid)
+ if not creature then
+ return
+ end
+
+ local target = Player(targetId)
+ if target then
+ combat:execute(creature, positionToVariant(target:getPosition()))
+ target:removeCondition(conditionOutfit)
+ end
+end
+
+function Creature:applyZoneEffect(var, combat, zoneName)
+ local outfitConfig = {
+ outfit = { lookType = 242, lookHead = 0, lookBody = 0, lookLegs = 0, lookFeet = 0, lookAddons = 0 },
+ time = 7000,
+ }
+
+ local zone = Zone.getByName(zoneName)
+ if not zone then
+ logger.error("Could not find zone '" .. zoneName .. "', you need use the 'BossLever' system")
+ return false
+ end
+
+ local target = zone:getRandomPlayer()
+ if not target then
+ return true
+ end
+
+ conditionOutfit:setTicks(outfitConfig.time)
+ conditionOutfit:setOutfit(outfitConfig.outfit)
+ target:addCondition(conditionOutfit)
+ target:getPosition():sendMagicEffect(CONST_ME_MAGIC_BLUE)
+
+ addEvent(delayedCastSpell, SoulWarQuest.goshnarsCrueltyWaveInterval * 1000, self:getId(), var, combat, target:getId())
+
+ return true
+end
+
+function string.toPosition(str)
+ local patterns = {
+ -- table format
+ "{%s*x%s*=%s*(%d+)%s*,%s*y%s*=%s*(%d+)%s*,%s*z%s*=%s*(%d+)%s*}",
+ -- Position format
+ "Position%s*%((%d+)%s*,%s*(%d+)%s*,%s*(%d+)%s*%)",
+ -- x, y, z format
+ "(%d+)%s*,%s*(%d+)%s*,%s*(%d+)",
+ }
+
+ for _, pattern in ipairs(patterns) do
+ local x, y, z = string.match(str, pattern)
+ if x and y and z then
+ return Position(tonumber(x), tonumber(y), tonumber(z))
+ end
+ end
+end
diff --git a/data-otservbr-global/monster/dragons/dragolisk.lua b/data-otservbr-global/monster/dragons/dragolisk.lua
new file mode 100644
index 00000000000..26cf05df8ec
--- /dev/null
+++ b/data-otservbr-global/monster/dragons/dragolisk.lua
@@ -0,0 +1,122 @@
+local mType = Game.createMonsterType("Dragolisk")
+local monster = {}
+
+monster.description = "a dragolisk"
+monster.experience = 5050
+monster.outfit = {
+ lookType = 1707,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.raceId = 2456
+monster.Bestiary = {
+ class = "Dragon",
+ race = BESTY_RACE_DRAGON,
+ toKill = 2500,
+ FirstUnlock = 100,
+ SecondUnlock = 1000,
+ CharmsPoints = 50,
+ Stars = 5,
+ Occurrence = 0,
+ Locations = "Nimmersatt's Breeding Ground",
+}
+
+monster.health = 6180
+monster.maxHealth = 6180
+monster.race = "blood"
+monster.corpse = 44654
+monster.speed = 165
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+ { text = "Just Look at me!", yell = false },
+ { text = "I'll stare you down", yell = false },
+ { text = "Let me have a look", yell = false },
+}
+
+monster.loot = {
+ { name = "Platinum Coin", chance = 52610, minCount = 1, maxCount = 46 },
+ { name = "Dragolisk Poison Gland", chance = 12600 },
+ { name = "Nimmersatt's Seal", chance = 8320 },
+ { id = 282, chance = 7580 },
+ { name = "Dragolisk Eye", chance = 5510 },
+ { name = "Green Gem", chance = 8260 },
+ { name = "Dragon's Tail", chance = 1003 },
+ { name = "Dragon Shield", chance = 400 },
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -200 },
+ { name = "combat", interval = 3000, chance = 27, type = COMBAT_PHYSICALDAMAGE, minDamage = -220, maxDamage = -250, length = 7, spread = 0, effect = CONST_ME_EXPLOSIONAREA, target = false },
+ { name = "combat", interval = 2500, chance = 25, type = COMBAT_PHYSICALDAMAGE, minDamage = -200, maxDamage = -250, radius = 4, effect = CONST_ME_POFF, target = true },
+ { name = "death chain", interval = 2500, chance = 20, minDamage = -250, maxDamage = -300, range = 7 },
+}
+
+monster.defenses = {
+ defense = 86,
+ armor = 86,
+ mitigation = 2.11,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 15 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = -10 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = -5 },
+ { type = COMBAT_HOLYDAMAGE, percent = -15 },
+ { type = COMBAT_DEATHDAMAGE, percent = -10 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/dragons/mega_dragon.lua b/data-otservbr-global/monster/dragons/mega_dragon.lua
new file mode 100644
index 00000000000..66696435486
--- /dev/null
+++ b/data-otservbr-global/monster/dragons/mega_dragon.lua
@@ -0,0 +1,126 @@
+local mType = Game.createMonsterType("Mega Dragon")
+local monster = {}
+
+monster.description = "a mega dragon"
+monster.experience = 7810
+monster.outfit = {
+ lookType = 1712,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.raceId = 2459
+monster.Bestiary = {
+ class = "Dragon",
+ race = BESTY_RACE_DRAGON,
+ toKill = 2500,
+ FirstUnlock = 100,
+ SecondUnlock = 1000,
+ CharmsPoints = 50,
+ Stars = 5,
+ Occurrence = 0,
+ Locations = "Nimmersatt's Breeding Ground",
+}
+
+monster.health = 7920
+monster.maxHealth = 7920
+monster.race = "blood"
+monster.corpse = 44663
+monster.speed = 170
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+ { text = "Just Look at me!", yell = false },
+ { text = "I'll stare you down", yell = false },
+ { text = "Let me have a look", yell = false },
+}
+
+monster.loot = {
+ { name = "Platinum Coin", chance = 50780, minCount = 1, maxCount = 48 },
+ { name = "Nimmersatt's Seal", chance = 8670 },
+ { id = 3039, chance = 10090 },
+ { name = "Molten Dragon Essence", chance = 4720 },
+ { name = "Prismatic Quartz", chance = 4460 },
+ { name = "Rainbow Quartz", chance = 4430, minCount = 1, maxCount = 2 },
+ { id = 3041, chance = 6790 },
+ { name = "Mega Dragon Heart", chance = 1850 },
+ { name = "Violet Gem", chance = 1470 },
+ { name = "Dragon Slayer", chance = 130 },
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -450 },
+ { name = "combat", interval = 2500, chance = 15, type = COMBAT_PHYSICALDAMAGE, minDamage = -250, maxDamage = -350, range = 2, effect = CONST_ME_BIG_SCRATCH, target = true },
+ { name = "combat", interval = 2500, chance = 20, type = COMBAT_FIREDAMAGE, minDamage = -250, maxDamage = -300, length = 8, spread = 4, effect = CONST_ME_EXPLOSIONHIT, target = false },
+ { name = "combat", interval = 2500, chance = 15, type = COMBAT_FIREDAMAGE, minDamage = -200, maxDamage = -300, range = 7, radius = 4, shootEffect = CONST_ANI_FIRE, effect = CONST_ME_FIREAREA, target = true },
+ { name = "combat", interval = 3000, chance = 5, type = COMBAT_PHYSICALDAMAGE, minDamage = -200, maxDamage = -350, radius = 4, effect = CONST_ME_POFF, target = true },
+ { name = "death chain", interval = 2500, chance = 15, minDamage = -200, maxDamage = -350, range = 7 },
+}
+
+monster.defenses = {
+ defense = 76,
+ armor = 76,
+ mitigation = 1.96,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = -10 },
+ { type = COMBAT_EARTHDAMAGE, percent = -10 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = -10 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/dragons/war_dragon.lua b/data-otservbr-global/monster/dragons/war_dragon.lua
new file mode 100644
index 00000000000..79d04bb2c0c
--- /dev/null
+++ b/data-otservbr-global/monster/dragons/war_dragon.lua
@@ -0,0 +1,125 @@
+local mType = Game.createMonsterType("Wardragon")
+local monster = {}
+
+monster.description = "a wardragon"
+monster.experience = 5810
+monster.outfit = {
+ lookType = 1708,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.raceId = 2458
+monster.Bestiary = {
+ class = "Dragon",
+ race = BESTY_RACE_DRAGON,
+ toKill = 2500,
+ FirstUnlock = 100,
+ SecondUnlock = 1000,
+ CharmsPoints = 50,
+ Stars = 5,
+ Occurrence = 0,
+ Locations = "Nimmersatt's Breeding Ground",
+}
+
+monster.health = 6960
+monster.maxHealth = 6960
+monster.race = "blood"
+monster.corpse = 44656
+monster.speed = 165
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+ { text = "Just Look at me!", yell = false },
+ { text = "I'll stare you down", yell = false },
+ { text = "Let me have a look", yell = false },
+}
+
+monster.loot = {
+ { name = "Platinum Coin", chance = 52560, minCount = 1, maxCount = 45 },
+ { name = "Wardragon Claw", chance = 14420 },
+ { name = "Nimmersatt's Seal", chance = 10620 },
+ { name = "Dragon Tongue", chance = 8450 },
+ { name = "Wardragon Tooth", chance = 6330 },
+ { name = "Gold Ingot", chance = 6007 },
+ { name = "Onyx Chip", chance = 4650, minCount = 1, maxCount = 2 },
+ { name = "Black Pearl", chance = 4000 },
+ { name = "White Gem", chance = 1000 },
+ { name = "Dragonbone Staff", chance = 740 },
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -450 },
+ { name = "combat", interval = 2500, chance = 20, type = COMBAT_DEATHDAMAGE, minDamage = -250, maxDamage = -400, radius = 4, effect = CONST_ME_MORTAREA, target = true },
+ { name = "combat", interval = 3000, chance = 25, type = COMBAT_FIREDAMAGE, minDamage = -300, maxDamage = -400, length = 8, spread = 4, effect = CONST_ME_EXPLOSIONHIT, target = false },
+ { name = "combat", interval = 3000, chance = 20, type = COMBAT_PHYSICALDAMAGE, minDamage = -275, maxDamage = -400, radius = 4, effect = CONST_ME_POFF, target = true },
+ { name = "combat", interval = 3000, chance = 15, type = COMBAT_PHYSICALDAMAGE, minDamage = -250, maxDamage = -300, range = 2, effect = CONST_ME_BIG_SCRATCH, target = true },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 80,
+ mitigation = 2.19,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = -5 },
+ { type = COMBAT_EARTHDAMAGE, percent = -10 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = -10 },
+ { type = COMBAT_HOLYDAMAGE, percent = -5 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/aspect_of_power.lua b/data-otservbr-global/monster/quests/soul_war/aspect_of_power.lua
index 8ff012c96ef..896cb520722 100644
--- a/data-otservbr-global/monster/quests/soul_war/aspect_of_power.lua
+++ b/data-otservbr-global/monster/quests/soul_war/aspect_of_power.lua
@@ -4,20 +4,20 @@ local monster = {}
monster.description = "an aspect of power"
monster.experience = 0
monster.outfit = {
- lookType = 1303,
+ lookType = 1306,
lookHead = 0,
lookBody = 0,
lookLegs = 0,
lookFeet = 0,
- lookAddons = 1,
+ lookAddons = 0,
lookMount = 0,
}
monster.health = 25000
monster.maxHealth = 25000
monster.race = "undead"
-monster.corpse = 0
-monster.speed = 125
+monster.corpse = 33949
+monster.speed = 175
monster.manaCost = 0
monster.changeTarget = {
@@ -25,8 +25,14 @@ monster.changeTarget = {
chance = 8,
}
+monster.events = {
+ "SoulWarAspectOfPowerDeath",
+}
+
monster.strategiesTarget = {
nearest = 100,
+ health = 20,
+ damage = 30,
}
monster.flags = {
@@ -64,16 +70,15 @@ monster.attacks = {
{ name = "combat", interval = 1700, chance = 15, type = COMBAT_EARTHDAMAGE, minDamage = -400, maxDamage = -950, radius = 3, shootEffect = CONST_ANI_ENVENOMEDARROW, effect = CONST_ME_HITBYPOISON, target = true },
{ name = "combat", interval = 1700, chance = 25, type = COMBAT_ENERGYDAMAGE, minDamage = -300, maxDamage = -850, length = 4, spread = 0, effect = CONST_ME_ENERGYHIT, target = false },
{ name = "combat", interval = 1700, chance = 35, type = COMBAT_DEATHDAMAGE, minDamage = -700, maxDamage = -1550, radius = 3, effect = CONST_ME_MORTAREA, target = false },
- { name = "outfit", interval = 1000, chance = 5, radius = 8, effect = CONST_ME_LOSEENERGY, target = false, duration = 5000, outfitMonster = "goshnar's hatred" },
- { name = "outfit", interval = 1000, chance = 5, radius = 8, effect = CONST_ME_LOSEENERGY, target = false, duration = 5000, outfitMonster = "goshnar's greed" },
- { name = "outfit", interval = 1000, chance = 5, radius = 8, effect = CONST_ME_LOSEENERGY, target = false, duration = 5000, outfitMonster = "goshnar's malice" },
- { name = "outfit", interval = 1000, chance = 5, radius = 8, effect = CONST_ME_LOSEENERGY, target = false, duration = 5000, outfitMonster = "goshnar's spite" },
}
monster.defenses = {
defense = 40,
- armor = 0,
- -- mitigation = ???,
+ armor = 40,
+ { name = "outfit", interval = 4000, chance = 30, target = false, duration = 4000, outfitMonster = "Goshnar's Malice" },
+ { name = "outfit", interval = 4000, chance = 30, target = false, duration = 4000, outfitMonster = "Goshnar's Cruelty" },
+ { name = "outfit", interval = 4000, chance = 30, target = false, duration = 4000, outfitMonster = "Goshnar's Hatred" },
+ { name = "outfit", interval = 4000, chance = 30, target = false, duration = 4000, outfitMonster = "Goshnar's Spite" },
}
monster.elements = {
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_megalomania.lua b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_blue.lua
similarity index 63%
rename from data-otservbr-global/monster/quests/soul_war/goshnars_megalomania.lua
rename to data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_blue.lua
index 7346e628cfc..1433d2ed7d4 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_megalomania.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_blue.lua
@@ -1,44 +1,40 @@
-local mType = Game.createMonsterType("Goshnar's Megalomania")
+local mType = Game.createMonsterType("Goshnar's Megalomania Blue")
local monster = {}
+monster.name = "Goshnar's Megalomania"
monster.description = "Goshnar's Megalomania"
-monster.experience = 200000
+monster.experience = 3000000
monster.outfit = {
- lookType = 1308,
- lookHead = 0,
- lookBody = 0,
- lookLegs = 0,
- lookFeet = 0,
- lookAddons = 0,
- lookMount = 0,
+ lookType = 1337,
}
-monster.events = {
- "SoulwarsBossDeath",
-}
-
-monster.health = 500000
-monster.maxHealth = 500000
+monster.health = 620000
+monster.maxHealth = 620000
monster.race = "undead"
monster.corpse = 33889
-monster.speed = 165
+monster.speed = 0
monster.manaCost = 0
-
-monster.changeTarget = {
- interval = 2000,
- chance = 10,
-}
+monster.maxSummons = 4
monster.bosstiary = {
bossRaceId = 1969,
bossRace = RARITY_NEMESIS,
}
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
monster.strategiesTarget = {
- nearest = 70,
+ nearest = 80,
health = 10,
damage = 10,
- random = 10,
+}
+
+monster.events = {
+ "GoshnarsHatredBuff",
+ "MegalomaniaDeath",
}
monster.flags = {
@@ -51,7 +47,7 @@ monster.flags = {
illusionable = false,
canPushItems = true,
canPushCreatures = true,
- staticAttackChance = 95,
+ staticAttackChance = 80,
targetDistance = 1,
runHealth = 0,
healthHidden = false,
@@ -67,14 +63,6 @@ monster.light = {
color = 0,
}
-monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 40, interval = 1000, count = 2 },
- { name = "aspect of power", chance = 50, interval = 1000, count = 2 },
- },
-}
-
monster.voices = {
interval = 5000,
chance = 10,
@@ -110,19 +98,14 @@ monster.loot = {
}
monster.attacks = {
- { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -8000 },
- { name = "combat", interval = 2000, chance = 30, type = COMBAT_PHYSICALDAMAGE, minDamage = -2950, maxDamage = -4400, range = 7, radius = 3, shootEffect = CONST_ANI_DEATH, effect = CONST_ME_MORTAREA, target = true },
- { name = "combat", interval = 2000, chance = 10, type = COMBAT_DEATHDAMAGE, minDamage = -3000, maxDamage = -5500, length = 8, spread = 0, effect = CONST_ME_INSECTS, target = false },
- { name = "singlecloudchain", interval = 6000, chance = 40, minDamage = -3300, maxDamage = -5500, range = 6, effect = CONST_ME_ENERGYHIT, target = true },
- { name = "combat", interval = 2000, chance = 10, type = COMBAT_DEATHDAMAGE, minDamage = -3300, maxDamage = -5200, length = 10, spread = 0, effect = CONST_ME_BLUE_GHOST, target = false },
+ { name = "melee", interval = 2000, chance = 100, minDamage = -400, maxDamage = -2225 },
+ { name = "megalomania blue", interval = 6000, chance = 100, target = true },
+ { name = "combat", interval = 30000, chance = 100, type = COMBAT_LIFEDRAIN, minDamage = -1000, maxDamage = -1500, length = 8, radius = 5, spread = 0, effect = CONST_ME_PINK_ENERGY_SPARK, target = true },
}
monster.defenses = {
- defense = 160,
- armor = 160,
- mitigation = 8.40,
- { name = "speed", interval = 1000, chance = 20, speedChange = 500, effect = CONST_ME_MAGIC_RED, target = false, duration = 10000 },
- { name = "combat", interval = 2000, chance = 25, type = COMBAT_HEALING, minDamage = 2250, maxDamage = 4250, effect = CONST_ME_MAGIC_BLUE, target = false },
+ defense = 55,
+ armor = 55,
}
monster.elements = {
@@ -145,18 +128,25 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
-mType.onThink = function(monster, interval) end
-
mType.onAppear = function(monster, creature)
if monster:getType():isRewardBoss() then
monster:setReward(true)
end
end
-mType.onDisappear = function(monster, creature) end
+local intervalBetweenExecutions = 10000
-mType.onMove = function(monster, creature, fromPosition, toPosition) end
+local zone = Zone.getByName("boss.goshnar's-megalomania-purple")
+local zonePositions = zone:getPositions()
-mType.onSay = function(monster, creature, type, message) end
+mType.onThink = function(monsterCallback, interval)
+ monsterCallback:onThinkGoshnarTormentCounter(interval, 36, intervalBetweenExecutions, SoulWarQuest.levers.goshnarsMegalomania.boss.position)
+ monsterCallback:onThinkMegalomaniaWhiteTiles(interval, zonePositions, 8000)
+ monsterCallback:goshnarsDefenseIncrease("cleansed-sanity-action")
+end
+
+mType.onDisappear = function(monster, creature)
+ creature:removeGoshnarsMegalomaniaMonsters(zone)
+end
mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_green.lua b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_green.lua
new file mode 100644
index 00000000000..f6d60b0b747
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_green.lua
@@ -0,0 +1,163 @@
+local mType = Game.createMonsterType("Goshnar's Megalomania Green")
+local monster = {}
+
+monster.name = "Goshnar's Megalomania"
+monster.description = "Goshnar's Megalomania"
+monster.experience = 3000000
+monster.outfit = {
+ lookType = 99,
+ lookHead = 95,
+ lookBody = 116,
+ lookLegs = 119,
+ lookFeet = 115,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.bosstiary = {
+ bossRaceId = 1969,
+ bossRace = RARITY_NEMESIS,
+}
+
+monster.health = 620000
+monster.maxHealth = 620000
+monster.race = "undead"
+monster.corpse = 33889
+monster.speed = 250
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.events = {
+ "GoshnarsHatredBuff",
+ "MegalomaniaDeath",
+}
+
+monster.strategiesTarget = {
+ nearest = 80,
+ health = 10,
+ damage = 10,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = true,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 80,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.loot = {
+ { name = "crystal coin", chance = 55000, minCount = 70, maxCount = 75 },
+ { id = 281, chance = 1150 }, -- giant shimmering pearl (green)
+ { name = "giant sapphire", chance = 10000, maxCount = 1 },
+ { name = "giant topaz", chance = 10000, maxCount = 1 },
+ { name = "violet gem", chance = 6000, maxCount = 1 },
+ { name = "blue gem", chance = 10000, maxCount = 3 },
+ { id = 3039, chance = 10000, maxCount = 3 }, -- red gem
+ { name = "green gem", chance = 10000, maxCount = 3 },
+ { name = "yellow gem", chance = 10000, maxCount = 3 },
+ { name = "white gem", chance = 6000, maxCount = 3 },
+ { name = "dragon figurine", chance = 10000, maxCount = 1 },
+ { name = "bullseye potion", chance = 15000, minCount = 10, maxCount = 25 },
+ { name = "mastermind potion", chance = 15000, minCount = 10, maxCount = 25 },
+ { name = "berserk potion", chance = 15000, minCount = 10, maxCount = 25 },
+ { name = "ultimate mana potion", chance = 18000, minCount = 50, maxCount = 100 },
+ { name = "supreme health potion", chance = 18000, minCount = 50, maxCount = 100 },
+ { name = "ultimate spirit potion", chance = 18000, minCount = 50, maxCount = 100 },
+ { name = "figurine of malice", chance = 400 },
+ { name = "figurine of cruelty", chance = 400 },
+ { name = "figurine of hatred", chance = 400 },
+ { name = "figurine of greed", chance = 400 },
+ { name = "figurine of spite", chance = 400 },
+ { name = "figurine of megalomania", chance = 400 },
+ { name = "megalomania's skull", chance = 400 },
+ { name = "megalomania's essence", chance = 400 },
+ { name = "bag you desire", chance = 100 },
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -800, maxDamage = -2500 },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1550, maxDamage = -2620, length = 8, spread = 0, effect = CONST_ME_MORTAREA, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1550, maxDamage = -2620, length = 8, spread = 0, effect = CONST_ME_BLACK_BLOOD, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1050, maxDamage = -2020, length = 8, spread = 3, effect = CONST_ME_GHOST_SMOKE, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1050, maxDamage = -2020, length = 8, spread = 3, effect = CONST_ME_SMALLCLOUDS, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -950, maxDamage = -1400, radius = 3, effect = CONST_ME_MORTAREA, target = true },
+ { name = "soulwars fear", interval = 35000, chance = 100, target = true },
+ { name = "megalomania transform elemental", interval = SoulWarQuest.goshnarsCrueltyWaveInterval * 1000, chance = 50 },
+ { name = "combat", interval = 30000, chance = 100, type = COMBAT_LIFEDRAIN, minDamage = -1000, maxDamage = -1500, length = 8, radius = 5, spread = 0, effect = CONST_ME_PINK_ENERGY_SPARK, target = true },
+}
+
+monster.defenses = {
+ defense = 55,
+ armor = 55,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType.onAppear = function(monster, creature)
+ if monster:getType():isRewardBoss() then
+ monster:setReward(true)
+ end
+end
+
+local intervalBetweenExecutions = 10000
+
+local zone = Zone.getByName("boss.goshnar's-megalomania-purple")
+local zonePositions = zone:getPositions()
+
+mType.onThink = function(monsterCallback, interval)
+ monsterCallback:onThinkGoshnarTormentCounter(interval, 36, intervalBetweenExecutions, SoulWarQuest.levers.goshnarsMegalomania.boss.position)
+ monsterCallback:onThinkMegalomaniaWhiteTiles(interval, zonePositions, 8000)
+ monsterCallback:goshnarsDefenseIncrease("cleansed-sanity-action")
+end
+
+mType.onDisappear = function(monster, creature)
+ creature:removeGoshnarsMegalomaniaMonsters(zone)
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_purple.lua b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_purple.lua
new file mode 100644
index 00000000000..79914af4935
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/goshnar's_megalomania_purple.lua
@@ -0,0 +1,129 @@
+local mType = Game.createMonsterType("Goshnar's Megalomania Purple")
+local monster = {}
+
+monster.name = "Goshnar's Megalomania"
+monster.description = "Goshnar's Megalomania"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1308,
+}
+
+monster.health = 620000
+monster.maxHealth = 620000
+monster.race = "undead"
+monster.corpse = 6028
+monster.speed = 250
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 80,
+ health = 10,
+ damage = 10,
+}
+
+monster.events = {
+ "GoshnarsHatredBuff",
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = true,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 80,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -800, maxDamage = -2500 },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1550, maxDamage = -2620, length = 8, spread = 0, effect = CONST_ME_MORTAREA, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1550, maxDamage = -2620, length = 8, spread = 0, effect = CONST_ME_BLACK_BLOOD, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1050, maxDamage = -2020, length = 8, spread = 3, effect = CONST_ME_GHOST_SMOKE, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -1050, maxDamage = -2020, length = 8, spread = 3, effect = CONST_ME_SMALLCLOUDS, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -950, maxDamage = -1400, radius = 3, effect = CONST_ME_MORTAREA, target = true },
+ { name = "soulwars fear", interval = 35000, chance = 100, target = true },
+ { name = "megalomania transform elemental", interval = SoulWarQuest.goshnarsCrueltyWaveInterval * 1000, chance = 50 },
+ { name = "combat", interval = 30000, chance = 100, type = COMBAT_LIFEDRAIN, minDamage = -1000, maxDamage = -1500, length = 8, radius = 5, spread = 0, effect = CONST_ME_PINK_ENERGY_SPARK, target = true },
+}
+
+monster.defenses = {
+ defense = 25,
+ armor = 25,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType.onSpawn = function(monster)
+ if monster:getType():isRewardBoss() then
+ monster:setReward(true)
+ end
+
+ if SoulWarQuest.changeBlueEvent then
+ stopEvent(SoulWarQuest.changeBlueEvent)
+ end
+ if SoulWarQuest.changePurpleEvent then
+ stopEvent(SoulWarQuest.changePurpleEvent)
+ end
+
+ local bossKV = monster:getSoulWarKV()
+ bossKV:set("aspect-of-power-death-count", 0)
+ monster:resetHatredDamageMultiplier()
+end
+
+local intervalBetweenExecutions = 10000
+
+local zone = Zone.getByName("boss.goshnar's-megalomania-purple")
+local zonePositions = zone:getPositions()
+
+mType.onThink = function(monsterCallback, interval)
+ monsterCallback:onThinkGoshnarTormentCounter(interval, 36, intervalBetweenExecutions, SoulWarQuest.levers.goshnarsMegalomania.boss.position)
+ monsterCallback:onThinkMegalomaniaWhiteTiles(interval, zonePositions, 8000)
+ monsterCallback:goshnarsDefenseIncrease("cleansed-sanity-action")
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_cruelty.lua b/data-otservbr-global/monster/quests/soul_war/goshnars_cruelty.lua
index 40957e0bde4..44f87cca574 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_cruelty.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnars_cruelty.lua
@@ -14,7 +14,8 @@ monster.outfit = {
}
monster.events = {
- "SoulwarsBossDeath",
+ "SoulWarBossesDeath",
+ "GoshnarsCrueltyBuff",
}
monster.health = 300000
@@ -67,14 +68,6 @@ monster.light = {
color = 0,
}
-monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 40, interval = 1000, count = 2 },
- { name = "mean maw", chance = 30, interval = 1000, count = 2 },
- },
-}
-
monster.voices = {
interval = 5000,
chance = 10,
@@ -112,6 +105,7 @@ monster.attacks = {
{ name = "singlecloudchain", interval = 6000, chance = 40, minDamage = -1700, maxDamage = -2500, range = 6, effect = CONST_ME_ENERGYHIT, target = true },
{ name = "combat", interval = 2000, chance = 30, type = COMBAT_PHYSICALDAMAGE, minDamage = -1000, maxDamage = -2500, range = 7, radius = 4, shootEffect = CONST_ANI_EXPLOSION, effect = CONST_ME_DRAWBLOOD, target = true },
{ name = "combat", interval = 2000, chance = 15, type = COMBAT_DEATHDAMAGE, minDamage = -1500, maxDamage = -3000, radius = 3, effect = CONST_ME_GROUNDSHAKER, target = false },
+ { name = "cruelty transform elemental", interval = SoulWarQuest.goshnarsCrueltyWaveInterval * 1000, chance = 50 },
}
monster.defenses = {
@@ -142,15 +136,33 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
-mType.onThink = function(monster, interval) end
+local firstTime = 0
+mType.onThink = function(monster, interval)
+ firstTime = firstTime + interval
+ -- Run only 15 seconds before creation
+ if firstTime >= 15000 then
+ monster:goshnarsDefenseIncrease("greedy-maw-action")
+ end
+end
+
+mType.onAppear = function(monster, creature) end
-mType.onAppear = function(monster, creature)
- if monster:getType():isRewardBoss() then
- monster:setReward(true)
+mType.onSpawn = function(monsterCallback)
+ if monsterCallback:getType():isRewardBoss() then
+ monsterCallback:setReward(true)
end
+
+ firstTime = 0
end
-mType.onDisappear = function(monster, creature) end
+mType.onDisappear = function(monster, creature)
+ if creature:getName() == "Goshnar's Cruelty" then
+ local eyeCreature = Creature("A Greedy Eye")
+ if eyeCreature then
+ eyeCreature:remove()
+ end
+ end
+end
mType.onMove = function(monster, creature, fromPosition, toPosition) end
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_greed.lua b/data-otservbr-global/monster/quests/soul_war/goshnars_greed.lua
index 171f963f5ff..f1f5284398e 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_greed.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnars_greed.lua
@@ -14,7 +14,7 @@ monster.outfit = {
}
monster.events = {
- "SoulwarsBossDeath",
+ "SoulWarBossesDeath",
}
monster.health = 300000
@@ -68,11 +68,7 @@ monster.light = {
}
monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 10, interval = 1000, count = 2 },
- { name = "hateful soul", chance = 10, interval = 1000, count = 2 },
- },
+ maxSummons = 1,
}
monster.voices = {
@@ -139,15 +135,56 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
-mType.onThink = function(monster, interval) end
+local immuneTimeCount = 0
+local isImmune = nil
+local createdSoulSphere = nil
+mType.onThink = function(monsterCallback, interval)
+ if GreedbeastKills >= 5 and isImmune == nil then
+ isImmune = monsterCallback:immune(false)
+ monsterCallback:teleportTo(Position(33741, 31659, 14))
+ monsterCallback:setSpeed(0)
+ createdSoulSphere = Game.createMonster("Soul Sphere", Position(33752, 31659, 14), true, true)
+ end
+ if isImmune ~= nil then
+ immuneTimeCount = immuneTimeCount + interval
+ logger.info("Immune time count {}", immuneTimeCount)
+ if immuneTimeCount >= 45000 then
+ monsterCallback:immune(true)
+ monsterCallback:setSpeed(monster.speed)
+ monsterCallback:teleportTo(Position(33746, 31666, 14))
+ immuneTimeCount = 0
+ GreedbeastKills = 0
+ isImmune = nil
+ if createdSoulSphere then
+ createdSoulSphere:remove()
+ end
+ end
+ end
+end
-mType.onAppear = function(monster, creature)
+mType.onSpawn = function(monster)
if monster:getType():isRewardBoss() then
monster:setReward(true)
end
+
+ isImmune = nil
+ monster:immune(true)
+ immuneTimeCount = 0
+ GreedbeastKills = 0
end
-mType.onDisappear = function(monster, creature) end
+mType.onDisappear = function(monster, creature)
+ if creature:getName() == "Greedbeast" then
+ logger.debug("GreedbeastKills {}", GreedbeastKills)
+ end
+ if creature:getName() == "Goshnar's Greed" then
+ logger.debug("Killed goshnar's greed")
+ if createdSoulSphere then
+ logger.debug("Found soul sphere, remove it")
+ createdSoulSphere:remove()
+ end
+ end
+end
mType.onMove = function(monster, creature, fromPosition, toPosition) end
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_hatred.lua b/data-otservbr-global/monster/quests/soul_war/goshnars_hatred.lua
index fa5ccf36984..47fedea4e7c 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_hatred.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnars_hatred.lua
@@ -14,7 +14,8 @@ monster.outfit = {
}
monster.events = {
- "SoulwarsBossDeath",
+ "GoshnarsHatredBuff",
+ "SoulWarBossesDeath",
}
monster.health = 300000
@@ -67,14 +68,6 @@ monster.light = {
color = 0,
}
-monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 10, interval = 1000, count = 2 },
- { name = "hateful soul", chance = 10, interval = 1000, count = 2 },
- },
-}
-
monster.voices = {
interval = 5000,
chance = 10,
@@ -143,13 +136,26 @@ monster.immunities = {
mType.onThink = function(monster, interval) end
-mType.onAppear = function(monster, creature)
+mType.onAppear = function(monster, creature) end
+
+mType.onSpawn = function(monster)
if monster:getType():isRewardBoss() then
monster:setReward(true)
end
+
+ monster:resetHatredDamageMultiplier()
end
-mType.onDisappear = function(monster, creature) end
+mType.onDisappear = function(monster, creature)
+ if creature:getName() == "Goshnar's Hatred" then
+ for _, monsterName in pairs(SoulWarQuest.burningHatredMonsters) do
+ local ashesCreature = Creature(monsterName)
+ if ashesCreature then
+ ashesCreature:remove()
+ end
+ end
+ end
+end
mType.onMove = function(monster, creature, fromPosition, toPosition) end
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_malice.lua b/data-otservbr-global/monster/quests/soul_war/goshnars_malice.lua
index 5e79dcccbdd..e0b054dc10d 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_malice.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnars_malice.lua
@@ -14,7 +14,8 @@ monster.outfit = {
}
monster.events = {
- "SoulwarsBossDeath",
+ "SoulWarBossesDeath",
+ "Goshnar's-Malice",
}
monster.health = 300000
@@ -67,14 +68,6 @@ monster.light = {
color = 0,
}
-monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 40, interval = 1000, count = 2 },
- { name = "malicious soul", chance = 30, interval = 1000, count = 2 },
- },
-}
-
monster.voices = {
interval = 5000,
chance = 10,
@@ -141,7 +134,19 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
-mType.onThink = function(monster, interval) end
+local zone = Zone.getByName("boss.goshnar's-malice")
+local zonePositions = zone:getPositions()
+
+local accumulatedTime = 0
+local desiredInterval = 40000
+mType.onThink = function(monster, interval)
+ accumulatedTime = accumulatedTime + interval
+ -- Execute only after 40 seconds
+ if accumulatedTime >= desiredInterval then
+ monster:createSoulWarWhiteTiles(SoulWarQuest.levers.goshnarsMalice.boss.position, zonePositions)
+ accumulatedTime = 0
+ end
+end
mType.onAppear = function(monster, creature)
if monster:getType():isRewardBoss() then
diff --git a/data-otservbr-global/monster/quests/soul_war/goshnars_spite.lua b/data-otservbr-global/monster/quests/soul_war/goshnars_spite.lua
index 19d35cd1af7..40817c335b0 100644
--- a/data-otservbr-global/monster/quests/soul_war/goshnars_spite.lua
+++ b/data-otservbr-global/monster/quests/soul_war/goshnars_spite.lua
@@ -14,7 +14,7 @@ monster.outfit = {
}
monster.events = {
- "SoulwarsBossDeath",
+ "SoulWarBossesDeath",
}
monster.health = 300000
@@ -67,14 +67,6 @@ monster.light = {
color = 0,
}
-monster.summon = {
- maxSummons = 4,
- summons = {
- { name = "dreadful harvester", chance = 40, interval = 1000, count = 2 },
- { name = "spiteful spitter", chance = 30, interval = 1000, count = 2 },
- },
-}
-
monster.voices = {
interval = 5000,
chance = 10,
@@ -111,6 +103,7 @@ monster.attacks = {
{ name = "singlecloudchain", interval = 6000, chance = 40, minDamage = -1700, maxDamage = -1900, range = 6, effect = CONST_ME_ENERGYHIT, target = true },
{ name = "combat", interval = 2000, chance = 30, type = COMBAT_EARTHDAMAGE, minDamage = -1200, maxDamage = -3500, range = 7, radius = 4, shootEffect = CONST_ANI_POISON, effect = CONST_ME_GREEN_RINGS, target = true },
{ name = "combat", interval = 2000, chance = 10, type = COMBAT_EARTHDAMAGE, minDamage = -1400, maxDamage = -2200, length = 8, spread = 0, effect = CONST_ME_GREEN_RINGS, target = false },
+ { name = "soulwars fear", interval = 2000, chance = 10, target = true },
}
monster.defenses = {
diff --git a/data-otservbr-global/monster/quests/soul_war/greedbeast.lua b/data-otservbr-global/monster/quests/soul_war/greedbeast.lua
new file mode 100644
index 00000000000..db7076d5710
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/greedbeast.lua
@@ -0,0 +1,98 @@
+local mType = Game.createMonsterType("Greedbeast")
+local monster = {}
+
+monster.description = "a greedbeast"
+monster.experience = 0
+monster.outfit = {
+ lookType = 101,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 10000
+monster.maxHealth = 10000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 200
+
+monster.events = {
+ "GreedMonsterDeath",
+}
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = false,
+ canWalkOnFire = false,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -500, condition = { type = CONDITION_POISON, totalDamage = 100, interval = 4000 } },
+ { name = "combat", interval = 2000, chance = 15, type = COMBAT_EARTHDAMAGE, minDamage = -50, maxDamage = -90, range = 7, shootEffect = CONST_ANI_POISON, effect = CONST_ME_POISONAREA, target = false },
+ { name = "combat", interval = 2000, chance = 10, type = COMBAT_LIFEDRAIN, minDamage = -25, maxDamage = -47, radius = 3, effect = CONST_ME_MAGIC_RED, target = false },
+ -- poison
+ { name = "condition", type = CONDITION_POISON, interval = 2000, chance = 10, minDamage = -200, maxDamage = -400, radius = 3, effect = CONST_ME_POISONAREA, target = false },
+ -- poison
+ { name = "condition", type = CONDITION_POISON, interval = 2000, chance = 10, minDamage = -200, maxDamage = -400, length = 6, spread = 0, effect = CONST_ME_POISONAREA, target = false },
+ { name = "speed", interval = 2000, chance = 15, speedChange = -600, target = true, duration = 13000 },
+}
+
+monster.defenses = {
+ defense = 70,
+ armor = 80,
+ mitigation = 1.15,
+ { name = "combat", interval = 2000, chance = 15, type = COMBAT_HEALING, minDamage = 50, maxDamage = 60, effect = CONST_ME_HITBYPOISON, target = false },
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = -10 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = -25 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/mirror_image.lua b/data-otservbr-global/monster/quests/soul_war/mirror_image.lua
index 8813ba2e93f..600550d3d78 100644
--- a/data-otservbr-global/monster/quests/soul_war/mirror_image.lua
+++ b/data-otservbr-global/monster/quests/soul_war/mirror_image.lua
@@ -13,6 +13,10 @@ monster.outfit = {
lookMount = 0,
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 35000
monster.maxHealth = 35000
monster.race = "blood"
@@ -106,4 +110,34 @@ monster.events = {
"MirrorImageTransform",
}
+mType.onPlayerAttack = function(monster, attackerPlayer)
+ logger.info("Player {}, attacking monster {}", attackerPlayer:getName(), monster:getName())
+
+ local apparitionType = ""
+
+ local sameVocationProbability = 70 -- 70% chance for create monster of first player attack vocation
+ if attackerPlayer:isDruid() then
+ apparitionType = "Druid's Apparition"
+ elseif attackerPlayer:isKnight() then
+ apparitionType = "Knight's Apparition"
+ elseif attackerPlayer:isPaladin() then
+ apparitionType = "Paladin's Apparition"
+ elseif attackerPlayer:isSorcerer() then
+ apparitionType = "Sorcerer's Apparition"
+ end
+
+ if math.random(100) > sameVocationProbability then
+ repeat
+ local randomIndex = math.random(#SoulWarQuest.apparitionNames)
+ if SoulWarQuest.apparitionNames[randomIndex] ~= apparitionType then
+ apparitionType = SoulWarQuest.apparitionNames[randomIndex]
+ break
+ end
+ until false
+ end
+
+ Game.createMonster(apparitionType, monster:getPosition(), true, true)
+ monster:remove()
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/undeads/bony_sea_devil.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/bony_sea_devil.lua
similarity index 95%
rename from data-otservbr-global/monster/undeads/bony_sea_devil.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/bony_sea_devil.lua
index 1199240df4b..d6e4bcfcafe 100644
--- a/data-otservbr-global/monster/undeads/bony_sea_devil.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/bony_sea_devil.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Ebb and Flow.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 24000
monster.maxHealth = 24000
monster.race = "undead"
@@ -96,7 +100,7 @@ monster.loot = {
{ name = "goblet of gloom", chance = 880 },
{ name = "glacier kilt", chance = 880 },
{ name = "glacial rod", chance = 1210 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
@@ -136,4 +140,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("Get out the way!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/demons/brachiodemon.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/brachiodemon.lua
similarity index 95%
rename from data-otservbr-global/monster/demons/brachiodemon.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/brachiodemon.lua
index 930ecfe469f..a76c9f40ee9 100644
--- a/data-otservbr-global/monster/demons/brachiodemon.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/brachiodemon.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Claustrophobic Inferno.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 25000
monster.maxHealth = 25000
monster.race = "blood"
@@ -99,7 +103,7 @@ monster.loot = {
{ name = "mastermind shield", chance = 420 },
{ name = "assassin dagger", chance = 340 },
{ name = "alloy legs", chance = 170 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
@@ -137,4 +141,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("Burn in hell!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/plants/branchy_crawler.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/branchy_crawler.lua
similarity index 94%
rename from data-otservbr-global/monster/plants/branchy_crawler.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/branchy_crawler.lua
index d319fcde5e5..f5dccd47eb5 100644
--- a/data-otservbr-global/monster/plants/branchy_crawler.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/branchy_crawler.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Rotten Wasteland.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 27000
monster.maxHealth = 27000
monster.race = "blood"
@@ -94,7 +98,7 @@ monster.loot = {
{ name = "twiceslicer", chance = 420 },
{ name = "crystalline sword", chance = 390 },
{ name = "ruthless axe", chance = 330 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
@@ -132,4 +136,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("My growth is your death!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/ashes_of_burning_hatred.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/ashes_of_burning_hatred.lua
new file mode 100644
index 00000000000..72c4082ee0e
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/ashes_of_burning_hatred.lua
@@ -0,0 +1,89 @@
+local mType = Game.createMonsterType("Ashes of Burning Hatred")
+local monster = {}
+
+monster.description = "a ashes of burning hatred"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 34009,
+}
+
+monster.health = 15000
+monster.maxHealth = 15000
+monster.race = "undead"
+monster.corpse = 5993
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 2000,
+ chance = 40,
+}
+
+monster.strategiesTarget = {
+ nearest = 0,
+ health = 0,
+ damage = 0,
+ random = 100,
+}
+
+monster.events = {
+ "BurningChangeForm",
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "combat", interval = 2000, chance = 25, type = COMBAT_FIREDAMAGE, minDamage = -555, maxDamage = -703, range = 9, shootEffect = CONST_ANI_FIRE, target = true },
+}
+
+monster.defenses = {
+ defense = 5,
+ armor = 10,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/blaze_of_burning_hatred.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/blaze_of_burning_hatred.lua
new file mode 100644
index 00000000000..c0b76ce98df
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/blaze_of_burning_hatred.lua
@@ -0,0 +1,89 @@
+local mType = Game.createMonsterType("Blaze of Burning Hatred")
+local monster = {}
+
+monster.description = "a blaze of burning hatred"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 34013,
+}
+
+monster.health = 15000
+monster.maxHealth = 15000
+monster.race = "undead"
+monster.corpse = 5993
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 2000,
+ chance = 40,
+}
+
+monster.strategiesTarget = {
+ nearest = 0,
+ health = 0,
+ damage = 0,
+ random = 100,
+}
+
+monster.events = {
+ "BurningChangeForm",
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "combat", interval = 2000, chance = 25, type = COMBAT_FIREDAMAGE, minDamage = -555, maxDamage = -703, range = 9, shootEffect = CONST_ANI_FIRE, target = true },
+}
+
+monster.defenses = {
+ defense = 5,
+ armor = 10,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/flame_of_burning_hatred.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/flame_of_burning_hatred.lua
new file mode 100644
index 00000000000..483c5156fb1
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/flame_of_burning_hatred.lua
@@ -0,0 +1,89 @@
+local mType = Game.createMonsterType("Flame of Burning Hatred")
+local monster = {}
+
+monster.description = "a flame of burning hatred"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 34011,
+}
+
+monster.health = 15000
+monster.maxHealth = 15000
+monster.race = "undead"
+monster.corpse = 5993
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 2000,
+ chance = 40,
+}
+
+monster.strategiesTarget = {
+ nearest = 0,
+ health = 0,
+ damage = 0,
+ random = 100,
+}
+
+monster.events = {
+ "BurningChangeForm",
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "combat", interval = 2000, chance = 25, type = COMBAT_FIREDAMAGE, minDamage = -555, maxDamage = -703, range = 9, shootEffect = CONST_ANI_FIRE, target = true },
+}
+
+monster.defenses = {
+ defense = 5,
+ armor = 10,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/spark_of_burning_hatred.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/spark_of_burning_hatred.lua
new file mode 100644
index 00000000000..dc9ff25f05e
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/spark_of_burning_hatred.lua
@@ -0,0 +1,89 @@
+local mType = Game.createMonsterType("Spark of Burning Hatred")
+local monster = {}
+
+monster.description = "a spark of burning hatred"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 34010,
+}
+
+monster.health = 15000
+monster.maxHealth = 15000
+monster.race = "undead"
+monster.corpse = 5993
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 2000,
+ chance = 40,
+}
+
+monster.strategiesTarget = {
+ nearest = 0,
+ health = 0,
+ damage = 0,
+ random = 100,
+}
+
+monster.events = {
+ "BurningChangeForm",
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "combat", interval = 2000, chance = 25, type = COMBAT_FIREDAMAGE, minDamage = -555, maxDamage = -703, range = 9, shootEffect = CONST_ANI_FIRE, target = true },
+}
+
+monster.defenses = {
+ defense = 5,
+ armor = 10,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/symbol_of_hatred.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/symbol_of_hatred.lua
new file mode 100644
index 00000000000..76d403fab1a
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/burning_hatred/symbol_of_hatred.lua
@@ -0,0 +1,84 @@
+local mType = Game.createMonsterType("Symbol of Hatred")
+local monster = {}
+
+monster.description = "a symbol of hatred"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 11427,
+}
+
+monster.health = 14000
+monster.maxHealth = 14000
+monster.race = "undead"
+monster.corpse = 33792
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = false,
+ canWalkOnFire = false,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.defenses = {
+ defense = 55,
+ armor = 55,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local intervalBetweenExecutions = 3000
+
+mType.onThink = function(monsterCallback, interval)
+ monsterCallback:onThinkGoshnarTormentCounter(interval, 30, intervalBetweenExecutions, SoulWarQuest.levers.goshnarsHatred.boss.position, "Goshnar's Hatred")
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/undeads/capricious_phantom.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/capricious_phantom.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/capricious_phantom.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/capricious_phantom.lua
index cd402486853..0b3bf5b8572 100644
--- a/data-otservbr-global/monster/undeads/capricious_phantom.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/capricious_phantom.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Ebb and Flow.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 30000
monster.maxHealth = 30000
monster.race = "undead"
@@ -93,7 +97,7 @@ monster.loot = {
{ id = 23542, chance = 1180 }, -- collar of blue plasma
{ name = "glacial rod", chance = 940 },
{ name = "ornate crossbow", chance = 940 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/distorted_phantom.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/distorted_phantom.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/distorted_phantom.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/distorted_phantom.lua
index c1fdc41461b..3f28aac16b9 100644
--- a/data-otservbr-global/monster/undeads/distorted_phantom.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/distorted_phantom.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Mirrored Nightmare.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 26000
monster.maxHealth = 26000
monster.race = "undead"
@@ -92,7 +96,7 @@ monster.loot = {
{ name = "spellbook of warding", chance = 2890 },
{ id = 23531, chance = 1930 }, -- ring of green plasma
{ name = "glacial rod", chance = 1290 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/druid's_apparition.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/druid's_apparition.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/druid's_apparition.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/druid's_apparition.lua
index 414de6155bf..957307c40eb 100644
--- a/data-otservbr-global/monster/undeads/druid's_apparition.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/druid's_apparition.lua
@@ -32,6 +32,10 @@ monster.corpse = 6081
monster.speed = 235
monster.manaCost = 0
+monster.events = {
+ "MirroredNightmareBossAccess",
+}
+
monster.changeTarget = {
interval = 4000,
chance = 0,
@@ -90,7 +94,7 @@ monster.loot = {
{ name = "platinum amulet", chance = 1750 },
{ name = "glacier robe", chance = 880 },
{ id = 23544, chance = 440 }, -- collar of red plasma
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/a_greedy_eye.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/a_greedy_eye.lua
new file mode 100644
index 00000000000..aed9dcfc829
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/a_greedy_eye.lua
@@ -0,0 +1,93 @@
+local mType = Game.createMonsterType("A Greedy Eye")
+local monster = {}
+
+monster.description = "a greedy eye"
+monster.experience = 0
+monster.outfit = {
+ lookType = 925,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 30000
+monster.maxHealth = 30000
+monster.race = "blood"
+monster.corpse = 5995
+monster.speed = 0
+monster.manaCost = 0
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 20,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 70,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.attacks = {
+ { name = "greedy eye beam", interval = 2000, chance = 100, minDamage = -1000, maxDamage = -1000 },
+}
+
+monster.defenses = {
+ defense = 55,
+ armor = 55,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/plants/cloak_of_terror.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/cloak_of_terror.lua
similarity index 94%
rename from data-otservbr-global/monster/plants/cloak_of_terror.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/cloak_of_terror.lua
index 184ba93fab2..ff25ab1f058 100644
--- a/data-otservbr-global/monster/plants/cloak_of_terror.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/cloak_of_terror.lua
@@ -26,6 +26,11 @@ monster.Bestiary = {
Locations = "Furious Crater.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+ "CloakOfTerrorHealthLoss",
+}
+
monster.health = 28000
monster.maxHealth = 28000
monster.race = "undead"
@@ -93,7 +98,7 @@ monster.loot = {
{ name = "blue gem", chance = 1490 },
{ name = "brooch of embracement", chance = 1490 },
{ name = "wand of defiance", chance = 990 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
@@ -131,4 +136,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("I am your terror!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/extra_dimensional/courage_leech.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/courage_leech.lua
similarity index 97%
rename from data-otservbr-global/monster/extra_dimensional/courage_leech.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/courage_leech.lua
index d8ad3147a8e..615e4637841 100644
--- a/data-otservbr-global/monster/extra_dimensional/courage_leech.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/courage_leech.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Furious Crater",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 27000
monster.maxHealth = 27000
monster.race = "undead"
@@ -92,7 +96,7 @@ monster.loot = {
{ name = "stone skin amulet", chance = 910 },
{ name = "nightmare blade", chance = 1190 },
{ name = "demonrage sword", chance = 600 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/poor_soul.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/poor_soul.lua
new file mode 100644
index 00000000000..316959ddcd8
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/poor_soul.lua
@@ -0,0 +1,118 @@
+local mType = Game.createMonsterType("Poor Soul")
+local monster = {}
+
+monster.description = "a poor soul"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1296,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 500
+monster.maxHealth = 500
+monster.race = "undead"
+monster.corpse = 33891
+monster.speed = 140
+monster.manaCost = 0
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 5,
+}
+
+monster.strategiesTarget = {
+ nearest = 60,
+ health = 10,
+ damage = 10,
+ random = 20,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+ { text = "I have a head start.", yell = false },
+ { text = "Look into my eyes! No, the other ones!", yell = false },
+ { text = "The mirrors can't contain the night!", yell = false },
+}
+
+monster.loot = {
+ { name = "crystal coin", chance = 70540 },
+ { name = "ultimate health potion", chance = 12220, maxCount = 7 },
+ { name = "violet gem", chance = 4560 },
+ { name = "green gem", chance = 5760 },
+ { name = "blue gem", chance = 4960 },
+ { name = "northwind rod", chance = 5920 },
+ { name = "sacred tree amulet", chance = 5520 },
+ { id = 33933, chance = 7920 }, -- apron
+ { id = 3067, chance = 7220 }, -- hailstorm rod
+ { name = "glacier shoes", chance = 2520 },
+ { name = "glacier robe", chance = 2220 },
+ { name = "stone skin amulet", chance = 5920 },
+ { id = 23533, chance = 4892 }, -- ring of red plasma
+ { id = 33932, chance = 3520 }, -- head
+ { name = "glacial rod", chance = 620 },
+ { id = 34024, chance = 650 }, -- gruesome fan
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -250, maxDamage = -450 },
+}
+
+monster.defenses = {
+ defense = 90,
+ armor = 105,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = -300 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = true },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/undeads/vibrant_phantom.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/vibrant_phantom.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/vibrant_phantom.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/vibrant_phantom.lua
index 9f17bee76ee..ae4c60df095 100644
--- a/data-otservbr-global/monster/undeads/vibrant_phantom.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/furious_crater/vibrant_phantom.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Furious Crater.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 27000
monster.maxHealth = 27000
monster.race = "undead"
@@ -94,7 +98,7 @@ monster.loot = {
{ name = "violet crystal shard", chance = 1080 },
{ id = 23529, chance = 1080 }, -- ring of blue plasma
{ name = "green gem", chance = 1080 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/quests/soul_war/hateful_soul.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/hateful_soul.lua
similarity index 99%
rename from data-otservbr-global/monster/quests/soul_war/hateful_soul.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/hateful_soul.lua
index 4aad05f3f33..69c413aa242 100644
--- a/data-otservbr-global/monster/quests/soul_war/hateful_soul.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/hateful_soul.lua
@@ -16,7 +16,7 @@ monster.outfit = {
monster.health = 25000
monster.maxHealth = 25000
monster.race = "undead"
-monster.corpse = 0
+monster.corpse = 33793
monster.speed = 125
monster.manaCost = 0
diff --git a/data-otservbr-global/monster/demons/infernal_demon.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_demon.lua
similarity index 97%
rename from data-otservbr-global/monster/demons/infernal_demon.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_demon.lua
index 7c4a1d80302..0ec107c192f 100644
--- a/data-otservbr-global/monster/demons/infernal_demon.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_demon.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Claustrophobic Inferno.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 32000
monster.maxHealth = 32000
monster.race = "blood"
@@ -94,7 +98,7 @@ monster.loot = {
{ name = "giant sword", chance = 2860 },
{ name = "magma boots", chance = 2290 },
{ name = "stone skin amulet", chance = 570 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/infernal_phantom.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_phantom.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/infernal_phantom.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_phantom.lua
index 0b61e291291..82b7fd2090a 100644
--- a/data-otservbr-global/monster/undeads/infernal_phantom.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/infernal_phantom.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Claustrophobic Inferno.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 26000
monster.maxHealth = 26000
monster.race = "undead"
@@ -95,7 +99,7 @@ monster.loot = {
{ name = "crystal mace", chance = 1610 },
{ name = "war axe", chance = 1410 },
{ name = "warrior's axe", chance = 1410 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/knight's_apparition.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/knight's_apparition.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/knight's_apparition.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/knight's_apparition.lua
index 0a4415aa2fd..f763e9b4dfe 100644
--- a/data-otservbr-global/monster/undeads/knight's_apparition.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/knight's_apparition.lua
@@ -33,6 +33,10 @@ monster.corpse = 111
monster.speed = 235
monster.manaCost = 0
+monster.events = {
+ "MirroredNightmareBossAccess",
+}
+
monster.changeTarget = {
interval = 4000,
chance = 0,
@@ -88,7 +92,7 @@ monster.loot = {
{ name = "giant sword", chance = 1720 },
{ name = "stone skin amulet", chance = 1500 },
{ name = "crown shield", chance = 640 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/demons/many_faces.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/many_faces.lua
similarity index 94%
rename from data-otservbr-global/monster/demons/many_faces.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/many_faces.lua
index 8b9724a6252..f60b999dbce 100644
--- a/data-otservbr-global/monster/demons/many_faces.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/many_faces.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Mirrored Nightmare.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 30000
monster.maxHealth = 30000
monster.race = "undead"
@@ -95,7 +99,7 @@ monster.loot = {
{ name = "glacier robe", chance = 2130 },
{ name = "gruesome fan", chance = 610 },
{ name = "glacial rod", chance = 610 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
@@ -133,4 +137,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("Hands off my comrades!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/greater_splinter_of_madness.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/greater_splinter_of_madness.lua
new file mode 100644
index 00000000000..2193aac5447
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/greater_splinter_of_madness.lua
@@ -0,0 +1,107 @@
+local mType = Game.createMonsterType("Greater Splinter of Madness")
+local monster = {}
+
+monster.description = "a greater splinter of madness"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1268,
+ lookHead = 0,
+ lookBody = 82,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 1,
+ lookMount = 0,
+}
+
+monster.health = 4000
+monster.maxHealth = 4000
+monster.race = "undead"
+monster.corpse = 32610
+monster.speed = 175
+monster.manaCost = 0
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 60,
+ health = 10,
+ damage = 10,
+ random = 20,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -100, maxDamage = -450 },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -350, maxDamage = -800, range = 4, shootEffect = CONST_ANI_SUDDENDEATH, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_ENERGYDAMAGE, minDamage = -300, maxDamage = -750, range = 4, shootEffect = CONST_ANI_ENERGY, target = true },
+}
+
+monster.defenses = {
+ defense = 40,
+ armor = 79,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType.onSpawn = function(monsterCallback)
+ addEvent(function(monsterId)
+ local eventMonster = Monster(monsterId)
+ if eventMonster then
+ eventMonster:setType("Mighty Splinter of Madness", true)
+ end
+ end, 120000, monsterCallback:getId())
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/lesser_splinter_of_madness.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/lesser_splinter_of_madness.lua
new file mode 100644
index 00000000000..7927ac18a26
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/lesser_splinter_of_madness.lua
@@ -0,0 +1,107 @@
+local mType = Game.createMonsterType("Lesser Splinter of Madness")
+local monster = {}
+
+monster.description = "a lesser splinter of madness"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1268,
+ lookHead = 0,
+ lookBody = 57,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 1,
+ lookMount = 0,
+}
+
+monster.health = 4000
+monster.maxHealth = 4000
+monster.race = "undead"
+monster.corpse = 32610
+monster.speed = 175
+monster.manaCost = 0
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 60,
+ health = 10,
+ damage = 10,
+ random = 20,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -100, maxDamage = -350 },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -300, maxDamage = -550, range = 4, shootEffect = CONST_ANI_SUDDENDEATH, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_ENERGYDAMAGE, minDamage = -300, maxDamage = -550, range = 4, shootEffect = CONST_ANI_ENERGY, target = true },
+}
+
+monster.defenses = {
+ defense = 40,
+ armor = 79,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType.onSpawn = function(monsterCallback)
+ addEvent(function(monsterId)
+ local eventMonster = Monster(monsterId)
+ if eventMonster then
+ eventMonster:setType("Greater Splinter of Madness", true)
+ end
+ end, 120000, monsterCallback:getId())
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/mighty_splinter_of_madness.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/mighty_splinter_of_madness.lua
new file mode 100644
index 00000000000..18dd5825b89
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/mighty_splinter_of_madness.lua
@@ -0,0 +1,113 @@
+local mType = Game.createMonsterType("Mighty Splinter of Madness")
+local monster = {}
+
+monster.description = "a mighty splinter of madness"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1268,
+ lookHead = 0,
+ lookBody = 93,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 1,
+ lookMount = 0,
+}
+
+monster.health = 4000
+monster.maxHealth = 4000
+monster.race = "undead"
+monster.corpse = 32610
+monster.speed = 175
+monster.manaCost = 0
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.strategiesTarget = {
+ nearest = 60,
+ health = 10,
+ damage = 10,
+ random = 20,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.voices = {
+ interval = 5000,
+ chance = 10,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -100, maxDamage = -750 },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_DEATHDAMAGE, minDamage = -450, maxDamage = -800, range = 4, shootEffect = CONST_ANI_SUDDENDEATH, target = false },
+ { name = "combat", interval = 2000, chance = 30, type = COMBAT_ENERGYDAMAGE, minDamage = -400, maxDamage = -800, range = 4, shootEffect = CONST_ANI_ENERGY, target = true },
+}
+
+monster.defenses = {
+ defense = 40,
+ armor = 79,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 100 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 100 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 100 },
+ { type = COMBAT_HOLYDAMAGE, percent = 100 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = true },
+ { type = "bleed", condition = false },
+}
+
+mType.onSpawn = function(monsterCallback)
+ addEvent(function(monsterId)
+ local eventMonster = Monster(monsterId)
+ if eventMonster then
+ creature:say("Goshnar's Megalomania feeds on its own madness and becomes stronger!", TALKTYPE_MONSTER_SAY, 0, 0, Position(34091, 31026, 9))
+ creature:remove()
+ local boss = Creature("Goshnar's Megalomania")
+ if boss then
+ boss:increaseHatredDamageMultiplier(5)
+ logger.debug("Goshnar's Megalomania has increased its damage multiplier to 5.")
+ end
+ end
+ end, 120000, monsterCallback:getId())
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/necromantic_focus.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/necromantic_focus.lua
new file mode 100644
index 00000000000..53c607ace7f
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/megalomania_room/necromantic_focus.lua
@@ -0,0 +1,82 @@
+local mType = Game.createMonsterType("Necromantic Focus")
+local monster = {}
+
+monster.description = "a necromantic focus"
+monster.experience = 0
+monster.outfit = {
+ lookTypeEx = 7059,
+}
+
+monster.health = 12000
+monster.maxHealth = 12000
+monster.race = "undead"
+monster.corpse = 33984
+monster.speed = 0
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.events = {
+ "NecromanticFocusDeath",
+}
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = false,
+ canWalkOnFire = false,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.defenses = {
+ defense = 55,
+ armor = 55,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/undeads/mould_phantom.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/mould_phantom.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/mould_phantom.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/mould_phantom.lua
index c3d7ea8ea86..1af17c877fa 100644
--- a/data-otservbr-global/monster/undeads/mould_phantom.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/mould_phantom.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Rotten Wasteland.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 28000
monster.maxHealth = 28000
monster.race = "undead"
@@ -93,7 +97,7 @@ monster.loot = {
{ id = 23529, chance = 1040 }, -- ring of blue plasma
{ name = "ornate crossbow", chance = 840 },
{ name = "crystal crossbow", chance = 620 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/paladin's_apparition.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/paladin's_apparition.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/paladin's_apparition.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/paladin's_apparition.lua
index 066266e70e3..2dc7a49555c 100644
--- a/data-otservbr-global/monster/undeads/paladin's_apparition.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/paladin's_apparition.lua
@@ -33,6 +33,10 @@ monster.corpse = 111
monster.speed = 235
monster.manaCost = 0
+monster.events = {
+ "MirroredNightmareBossAccess",
+}
+
monster.changeTarget = {
interval = 4000,
chance = 0,
@@ -91,7 +95,7 @@ monster.loot = {
{ name = "stone skin amulet", chance = 1560 },
{ id = 23542, chance = 1250 }, -- collar of blue plasma
{ id = 23529, chance = 1250 }, -- ring of blue plasma
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/constructs/rotten_golem.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/rotten_golem.lua
similarity index 97%
rename from data-otservbr-global/monster/constructs/rotten_golem.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/rotten_golem.lua
index edc6f909a70..f5d5808b05c 100644
--- a/data-otservbr-global/monster/constructs/rotten_golem.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/rotten_golem.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Rotten Wasteland.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 28000
monster.maxHealth = 28000
monster.race = "venom"
@@ -91,7 +95,7 @@ monster.loot = {
{ name = "stone skin amulet", chance = 740 },
{ name = "terra mantle", chance = 510 },
{ name = "rubber cap", chance = 430 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/undeads/sorcerer's_apparition.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/sorcerer's_apparition.lua
similarity index 97%
rename from data-otservbr-global/monster/undeads/sorcerer's_apparition.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/sorcerer's_apparition.lua
index 8cf059be62c..e554c27dfd9 100644
--- a/data-otservbr-global/monster/undeads/sorcerer's_apparition.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/sorcerer's_apparition.lua
@@ -33,6 +33,10 @@ monster.corpse = 6081
monster.speed = 235
monster.manaCost = 0
+monster.events = {
+ "MirroredNightmareBossAccess",
+}
+
monster.changeTarget = {
interval = 4000,
chance = 0,
@@ -92,7 +96,7 @@ monster.loot = {
{ name = "wand of starstorm", chance = 1310 },
{ name = "stone skin amulet", chance = 1310 },
{ name = "alloy legs", chance = 440 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/elementals/turbulent_elemental.lua b/data-otservbr-global/monster/quests/soul_war/normal_monsters/turbulent_elemental.lua
similarity index 97%
rename from data-otservbr-global/monster/elementals/turbulent_elemental.lua
rename to data-otservbr-global/monster/quests/soul_war/normal_monsters/turbulent_elemental.lua
index 3bef5da44e3..6d7804f2207 100644
--- a/data-otservbr-global/monster/elementals/turbulent_elemental.lua
+++ b/data-otservbr-global/monster/quests/soul_war/normal_monsters/turbulent_elemental.lua
@@ -26,6 +26,10 @@ monster.Bestiary = {
Locations = "Ebb and Flow.",
}
+monster.events = {
+ "FourthTaintBossesPrepareDeath",
+}
+
monster.health = 28000
monster.maxHealth = 28000
monster.race = "blood"
@@ -91,7 +95,7 @@ monster.loot = {
{ name = "crystalline armor", chance = 710 },
{ name = "rubber cap", chance = 710 },
{ name = "stone skin amulet", chance = 470 },
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/monster/quests/soul_war/powerful_soul.lua b/data-otservbr-global/monster/quests/soul_war/powerful_soul.lua
new file mode 100644
index 00000000000..0ece602eab5
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/powerful_soul.lua
@@ -0,0 +1,115 @@
+local mType = Game.createMonsterType("Powerful Soul")
+local monster = {}
+
+monster.description = "a powerful soul"
+monster.experience = 0
+monster.outfit = {
+ lookType = 568,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 30000
+monster.maxHealth = 30000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 80
+monster.manaCost = 0
+
+monster.events = {
+ "GreedMonsterDeath",
+}
+
+monster.changeTarget = {
+ interval = 1000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = false,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -2000, maxDamage = -3000 },
+ { name = "combat", interval = 2000, chance = 15, type = COMBAT_LIFEDRAIN, minDamage = -2000, maxDamage = -3000, range = 1, effect = CONST_ME_MAGIC_RED, target = false },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 90,
+ mitigation = 2,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local transformTimeCount = 0
+mType.onThink = function(monster, interval)
+ transformTimeCount = transformTimeCount + interval
+ if transformTimeCount == 8000 then
+ CreateGoshnarsGreedMonster("Weak Soul", GreedMonsters[monster:getName()])
+ monster:remove()
+ local boss = Creature("Goshnar's Greed")
+ if boss then
+ for elementType, reflectPercent in pairs(SoulWarReflectDamageMap) do
+ boss:addReflectElement(elementType, reflectPercent)
+ end
+ boss:addDefense(10)
+ boss:setMaxHealth(boss:getMaxHealth() + 10000)
+ boss:addHealth(10000)
+ end
+ transformTimeCount = 0
+ end
+end
+
+mType.onSpawn = function(monster)
+ transformTimeCount = 0
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/soul_cage.lua b/data-otservbr-global/monster/quests/soul_war/soul_cage.lua
new file mode 100644
index 00000000000..c6c65b32547
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/soul_cage.lua
@@ -0,0 +1,71 @@
+local mType = Game.createMonsterType("Soul Cage")
+local monster = {}
+
+monster.description = "a soul cage"
+monster.experience = 100000
+monster.outfit = {
+ lookType = 863,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 100000
+monster.maxHealth = 100000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 0
+monster.manaCost = 0
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+}
+
+monster.events = {
+ "SoulCageDeath",
+ "SoulCageHealthChange",
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 100,
+ { name = "Heal Malice", interval = 2000, chance = 90, target = false },
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "invisible", condition = true },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/soul_sphere.lua b/data-otservbr-global/monster/quests/soul_war/soul_sphere.lua
new file mode 100644
index 00000000000..160879ba9db
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/soul_sphere.lua
@@ -0,0 +1,122 @@
+local mType = Game.createMonsterType("Soul Sphere")
+local monster = {}
+
+monster.description = "a soul sphere"
+monster.experience = 0
+monster.outfit = {
+ lookType = 979,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 10000
+monster.maxHealth = 10000
+monster.corpse = 0
+monster.speed = 0
+monster.manaCost = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -500, maxDamage = -1000 },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 90,
+ mitigation = 0.51,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 100 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local moveTimeCount = 0
+local stop = false
+mType.onThink = function(monster, interval)
+ if stop then
+ return
+ end
+
+ moveTimeCount = moveTimeCount + interval
+ if moveTimeCount >= 3000 then
+ local currentPos = monster:getPosition()
+ local newPos = Position(currentPos.x - 1, currentPos.y, currentPos.z)
+
+ local nextTile = Tile(newPos)
+ if nextTile then
+ for _, creatureId in pairs(nextTile:getCreatures()) do
+ local tileMonster = Monster(creatureId)
+ if tileMonster and tileMonster:getName() == "Goshnar's Greed" then
+ tileMonster:setHealth(tileMonster:getMaxHealth())
+ stop = true
+ return
+ end
+ end
+ end
+
+ if not stop then
+ monster:teleportTo(newPos, true)
+ moveTimeCount = 0
+ end
+ end
+end
+
+mType.onSpawn = function(monster)
+ moveTimeCount = 0
+ stop = false
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/soulsnatcher.lua b/data-otservbr-global/monster/quests/soul_war/soulsnatcher.lua
new file mode 100644
index 00000000000..faa819dec6e
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/soulsnatcher.lua
@@ -0,0 +1,115 @@
+local mType = Game.createMonsterType("Soulsnatcher")
+local monster = {}
+
+monster.description = "a soulsnatcher"
+monster.experience = 0
+monster.outfit = {
+ lookType = 1268,
+ lookHead = 0,
+ lookBody = 94,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 10000
+monster.maxHealth = 10000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 80
+monster.manaCost = 0
+
+monster.events = {
+ "GreedMonsterDeath",
+}
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = true,
+ canPushItems = true,
+ canPushCreatures = false,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -1000, maxDamage = -1500 },
+ { name = "soulsnatcher-lifedrain-beam", interval = 2000, chance = 20, minDamage = -1000, maxDamage = -1500, target = false },
+ { name = "soulsnatcher-lifedrain-missile", interval = 2000, chance = 25, minDamage = -1000, maxDamage = -1500, target = true },
+ { name = "soulsnatcher-manadrain-ball", interval = 2000, chance = 30, minDamage = -500, maxDamage = -1000 },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 90,
+ mitigation = 0.51,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 0 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local transformTimeCount = 0
+mType.onThink = function(monster, interval)
+ transformTimeCount = transformTimeCount + interval
+ if transformTimeCount == 8000 then
+ transformTimeCount = 0
+ local zone = Zone("boss.goshnar's-greed")
+ if zone then
+ local players = zone:getPlayers()
+ for _, player in ipairs(players) do
+ player:addHealth(-math.random(500, 1000))
+ end
+ end
+ CreateGoshnarsGreedMonster(monster:getName(), GreedMonsters[monster:getName()])
+ monster:remove()
+ end
+end
+
+mType.onSpawn = function(monster)
+ transformTimeCount = 0
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/spiteful_spitter.lua b/data-otservbr-global/monster/quests/soul_war/spiteful_spitter.lua
index 8b8cec55ccf..d51b52e2e84 100644
--- a/data-otservbr-global/monster/quests/soul_war/spiteful_spitter.lua
+++ b/data-otservbr-global/monster/quests/soul_war/spiteful_spitter.lua
@@ -99,4 +99,8 @@ monster.immunities = {
{ type = "bleed", condition = false },
}
+mType.onThink = function(monster, interval)
+ monster:tryTeleportToPlayer("You have been chosen for a harvest!")
+end
+
mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/strong_soul.lua b/data-otservbr-global/monster/quests/soul_war/strong_soul.lua
new file mode 100644
index 00000000000..ae3615b6eb4
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/strong_soul.lua
@@ -0,0 +1,106 @@
+local mType = Game.createMonsterType("Strong Soul")
+local monster = {}
+
+monster.description = "a strong soul"
+monster.experience = 0
+monster.outfit = {
+ lookType = 566,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 20000
+monster.maxHealth = 20000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 80
+monster.manaCost = 0
+
+monster.events = {
+ "GreedMonsterDeath",
+}
+
+monster.changeTarget = {
+ interval = 1000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = false,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -1000, maxDamage = -2000 },
+ { name = "combat", interval = 2000, chance = 15, type = COMBAT_LIFEDRAIN, minDamage = -1000, maxDamage = -2000, range = 1, effect = CONST_ME_MAGIC_RED, target = false },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 90,
+ mitigation = 2,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local transformTimeCount = 0
+mType.onThink = function(monster, interval)
+ transformTimeCount = transformTimeCount + interval
+ if transformTimeCount == 8000 then
+ Game.createMonster("Powerful Soul", GreedMonsters[monster:getName()], true, false)
+ monster:remove()
+ transformTimeCount = 0
+ end
+end
+
+mType.onSpawn = function(monster)
+ transformTimeCount = 0
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/weak_soul.lua b/data-otservbr-global/monster/quests/soul_war/weak_soul.lua
new file mode 100644
index 00000000000..b7150efa00f
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/weak_soul.lua
@@ -0,0 +1,106 @@
+local mType = Game.createMonsterType("Weak Soul")
+local monster = {}
+
+monster.description = "a weak soul"
+monster.experience = 0
+monster.outfit = {
+ lookType = 48,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 10000
+monster.maxHealth = 10000
+monster.race = "undead"
+monster.corpse = 0
+monster.speed = 80
+monster.manaCost = 0
+
+monster.events = {
+ "GreedMonsterDeath",
+}
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 0,
+}
+
+monster.strategiesTarget = {
+ nearest = 100,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = false,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = 0, maxDamage = -1000 },
+ { name = "combat", interval = 2000, chance = 15, type = COMBAT_LIFEDRAIN, minDamage = -500, maxDamage = -1000, range = 1, effect = CONST_ME_MAGIC_RED, target = false },
+}
+
+monster.defenses = {
+ defense = 80,
+ armor = 90,
+ mitigation = 0.51,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 100 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 100 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 100 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 100 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 100 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+local transformTimeCount = 0
+mType.onThink = function(monster, interval)
+ transformTimeCount = transformTimeCount + interval
+ if transformTimeCount == 8000 then
+ Game.createMonster("Strong Soul", GreedMonsters[monster:getName()], true, false)
+ monster:remove()
+ transformTimeCount = 0
+ end
+end
+
+mType.onSpawn = function(monster)
+ transformTimeCount = 0
+end
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/quests/soul_war/weeping_soul.lua b/data-otservbr-global/monster/quests/soul_war/weeping_soul.lua
new file mode 100644
index 00000000000..3631fa3cb8e
--- /dev/null
+++ b/data-otservbr-global/monster/quests/soul_war/weeping_soul.lua
@@ -0,0 +1,91 @@
+local mType = Game.createMonsterType("Weeping Soul")
+local monster = {}
+
+monster.description = "a weeping soul"
+monster.experience = 0
+monster.outfit = {
+ lookType = 48,
+ lookHead = 0,
+ lookBody = 0,
+ lookLegs = 0,
+ lookFeet = 0,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+monster.health = 7000
+monster.maxHealth = 7000
+monster.race = "undead"
+monster.corpse = 33876
+monster.speed = 150
+monster.manaCost = 100
+monster.maxSummons = 0
+
+monster.changeTarget = {
+ interval = 4000,
+ chance = 15,
+}
+
+monster.strategiesTarget = {
+ nearest = 60,
+ health = 10,
+ damage = 10,
+ random = 20,
+}
+
+monster.flags = {
+ summonable = false,
+ attackable = true,
+ hostile = true,
+ convinceable = false,
+ pushable = false,
+ rewardBoss = false,
+ illusionable = false,
+ canPushItems = true,
+ canPushCreatures = true,
+ staticAttackChance = 90,
+ targetDistance = 1,
+ runHealth = 0,
+ healthHidden = false,
+ isBlockable = false,
+ canWalkOnEnergy = true,
+ canWalkOnFire = true,
+ canWalkOnPoison = true,
+ pet = false,
+}
+
+monster.light = {
+ level = 0,
+ color = 0,
+}
+
+monster.attacks = {
+ { name = "melee", interval = 2000, chance = 100, minDamage = -220, maxDamage = -650 },
+}
+
+monster.defenses = {
+ defense = 55,
+ armor = 55,
+}
+
+monster.elements = {
+ { type = COMBAT_PHYSICALDAMAGE, percent = 40 },
+ { type = COMBAT_ENERGYDAMAGE, percent = 0 },
+ { type = COMBAT_EARTHDAMAGE, percent = 0 },
+ { type = COMBAT_FIREDAMAGE, percent = 0 },
+ { type = COMBAT_LIFEDRAIN, percent = 0 },
+ { type = COMBAT_MANADRAIN, percent = 0 },
+ { type = COMBAT_DROWNDAMAGE, percent = 0 },
+ { type = COMBAT_ICEDAMAGE, percent = 0 },
+ { type = COMBAT_HOLYDAMAGE, percent = 0 },
+ { type = COMBAT_DEATHDAMAGE, percent = 0 },
+}
+
+monster.immunities = {
+ { type = "paralyze", condition = true },
+ { type = "outfit", condition = false },
+ { type = "invisible", condition = false },
+ { type = "bleed", condition = false },
+}
+
+mType:register(monster)
diff --git a/data-otservbr-global/monster/reptiles/two-headed_turtle.lua b/data-otservbr-global/monster/reptiles/two-headed_turtle.lua
index 2d1c2090b37..50b49aa8cc5 100644
--- a/data-otservbr-global/monster/reptiles/two-headed_turtle.lua
+++ b/data-otservbr-global/monster/reptiles/two-headed_turtle.lua
@@ -70,7 +70,7 @@ monster.loot = {
{ name = "two-headed turtle heads", chance = 8700 },
{ name = "strong mana potion", chance = 13373 },
{ name = "hydrophytes", chance = 11000 },
- { id = 1047, chance = 6388 }, -- bone
+ { id = 3115, chance = 6388 }, -- bone
{ name = "glacier shoes", chance = 4650 },
{ id = 281, chance = 3582 }, -- giant shimmering pearl (green)
{ name = "small tropical fish", chance = 3582 },
diff --git a/data-otservbr-global/monster/undeads/hazardous_phantom.lua b/data-otservbr-global/monster/undeads/hazardous_phantom.lua
index 29384eafe55..1cd1eb7d38c 100644
--- a/data-otservbr-global/monster/undeads/hazardous_phantom.lua
+++ b/data-otservbr-global/monster/undeads/hazardous_phantom.lua
@@ -20,6 +20,10 @@ monster.corpse = 34125
monster.speed = 100
monster.manaCost = 0
+monster.events = {
+ "HazardousPhantomDeath",
+}
+
monster.changeTarget = {
interval = 4000,
chance = 0,
@@ -78,7 +82,7 @@ monster.loot = {
{ id = 282, chance = 1570 }, -- giant shimmering pearl
{ name = "wand of everblazing", chance = 790 },
{ id = 23542, chance = 790 }, -- collar of blue plasma
- { id = 34109, chance = 20 }, -- bag you desire
+ { name = "bag you desire", chance = 15 },
}
monster.attacks = {
diff --git a/data-otservbr-global/npc/arito.lua b/data-otservbr-global/npc/arito.lua
index caa33f8348f..1eab9a186c3 100644
--- a/data-otservbr-global/npc/arito.lua
+++ b/data-otservbr-global/npc/arito.lua
@@ -134,9 +134,6 @@ npcConfig.shop = {
{ itemName = "mug of lemonade", clientId = 2880, buy = 2, count = 12 },
{ itemName = "mug of water", clientId = 2880, buy = 1, count = 1 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 1, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/armenius.lua b/data-otservbr-global/npc/armenius.lua
index 8b24dc1d419..9164a78240e 100644
--- a/data-otservbr-global/npc/armenius.lua
+++ b/data-otservbr-global/npc/armenius.lua
@@ -97,8 +97,6 @@ npcConfig.shop = {
{ itemName = "mug of rum", clientId = 2880, buy = 10, count = 13 },
{ itemName = "mug of wine", clientId = 2880, buy = 4, count = 2 },
{ itemName = "tomato", clientId = 3596, buy = 3 },
- { itemName = "vial of beer", clientId = 2874, buy = 3, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 4, count = 1, subType = 2 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/bolfona.lua b/data-otservbr-global/npc/bolfona.lua
index 83ee91f7ac9..adf900568ff 100644
--- a/data-otservbr-global/npc/bolfona.lua
+++ b/data-otservbr-global/npc/bolfona.lua
@@ -103,7 +103,6 @@ npcConfig.shop = {
{ itemName = "ham", clientId = 3582, buy = 8 },
{ itemName = "meat", clientId = 3577, buy = 5 },
{ itemName = "mug of beer", clientId = 2880, buy = 2, count = 3 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/boozer.lua b/data-otservbr-global/npc/boozer.lua
index a2eaa22d179..b63fa7620fb 100644
--- a/data-otservbr-global/npc/boozer.lua
+++ b/data-otservbr-global/npc/boozer.lua
@@ -98,8 +98,6 @@ npcConfig.shop = {
{ itemName = "mug of lemonade", clientId = 2880, buy = 2, count = 12 },
{ itemName = "mug of water", clientId = 2880, buy = 2, count = 1 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/clyde.lua b/data-otservbr-global/npc/clyde.lua
index 4142cee2cbc..fcaf9ea011a 100644
--- a/data-otservbr-global/npc/clyde.lua
+++ b/data-otservbr-global/npc/clyde.lua
@@ -64,9 +64,6 @@ npcConfig.shop = {
{ itemName = "mug of water", clientId = 2880, buy = 2, count = 1 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
{ itemName = "tomato", clientId = 3596, buy = 3 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 1, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/dane.lua b/data-otservbr-global/npc/dane.lua
index f2dd71762c0..c2e3c612414 100644
--- a/data-otservbr-global/npc/dane.lua
+++ b/data-otservbr-global/npc/dane.lua
@@ -67,8 +67,6 @@ npcConfig.shop = {
{ itemName = "bottle of lemonade", clientId = 2875, buy = 12, count = 12 },
{ itemName = "bottle of milk", clientId = 2875, buy = 4, count = 9 },
{ itemName = "bottle of water", clientId = 2875, buy = 2, count = 1 },
- { itemName = "vial of milk", clientId = 2874, buy = 4, count = 1, subType = 9 },
- { itemName = "vial of water", clientId = 2874, buy = 2, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/flickering_soul.lua b/data-otservbr-global/npc/flickering_soul.lua
new file mode 100644
index 00000000000..2010198da6e
--- /dev/null
+++ b/data-otservbr-global/npc/flickering_soul.lua
@@ -0,0 +1,203 @@
+local internalNpcName = "Flickering Soul"
+local npcType = Game.createNpcType(internalNpcName)
+local npcConfig = {}
+
+npcConfig.name = internalNpcName
+npcConfig.description = internalNpcName
+
+npcConfig.health = 100
+npcConfig.maxHealth = npcConfig.health
+npcConfig.walkInterval = 2000
+npcConfig.walkRadius = 2
+
+npcConfig.outfit = {
+ lookType = 1219,
+ lookHead = 6,
+ lookBody = 26,
+ lookLegs = 26,
+ lookFeet = 6,
+ lookAddons = 0,
+ lookMount = 0,
+}
+
+npcConfig.flags = {
+ floorchange = false,
+}
+
+local keywordHandler = KeywordHandler:new()
+local npcHandler = NpcHandler:new(keywordHandler)
+
+npcType.onThink = function(npc, interval)
+ npcHandler:onThink(npc, interval)
+end
+
+npcType.onAppear = function(npc, player)
+ npcHandler:onAppear(npc, player)
+end
+
+npcType.onDisappear = function(npc, player)
+ npcHandler:onDisappear(npc, player)
+end
+
+npcType.onMove = function(npc, player, fromPosition, toPosition)
+ npcHandler:onMove(npc, player, fromPosition, toPosition)
+end
+
+npcType.onSay = function(npc, player, type, message)
+ npcHandler:onSay(npc, player, type, message)
+end
+
+npcType.onCloseChannel = function(npc, player)
+ npcHandler:onCloseChannel(npc, player)
+end
+
+local function playerSayCallback(npc, player, type, message)
+ if not npcHandler:checkInteraction(npc, player) then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+
+ local playerId = player:getId()
+ if MsgContains(message, "living") then
+ npcHandler:say("It has been a while since I roamed the world of the living in a mortal shell.", npc, player)
+ elseif MsgContains(message, "mortal") then
+ npcHandler:say("I had many names in my live. The one that would be the most recognizable is probably the name Goshnar. Even that was an assumed name that I took when I left my mundane past behind.", npc, player)
+ elseif MsgContains(message, "Goshnar") then
+ npcHandler:say({
+ "I was once known as the necromant king. ...",
+ "For some it was meant as a curse, others used the name with reverence. To me it was just another stepping stone, in a life that burned with ambition.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "ambition") then
+ npcHandler:say({
+ "My ambitions were high and knew no limits. Mastery over life and death was but a milestone that I wanted to accomplish. In the end I aspired probably somewhat like godhood. ...",
+ "Though in hindsight even that wouldn't have been enough. There was a hunger in me that nothing could put to rest.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "milestone") then
+ npcHandler:say("Everything in my life was just a tool to further my goals. The brotherhood of bones was just a tool for me. As was everyone or everything. In the path I had chosen nothing mattered but me and my ambitions.", npc, player)
+ elseif MsgContains(message, "everything") then
+ npcHandler:say("Necromancy was a passion at first, another tool while I amassed power and a crutch when my ambitions surpassed that what it could accomplish.", npc, player)
+ elseif MsgContains(message, "accomplish") then
+ npcHandler:say({
+ "I was so convinced about my brilliance, my greatness, my destiny. And this hunger for more, it let me not have peace at any point in my life. I was always driven. There was no time to rest. ...",
+ "And there was no looking back. I never cared to remember my humble beginnings, what I had sacrificed to get where I was. All that I had left behind and that I had lost forever. ...",
+ "Now I see the bitter irony. I could bring back the dead, but I couldn't create second chances. I couldn't restore the truly important things that I had lost.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "dead") then
+ npcHandler:say("My demise did not meet me unprepared. As a powerful necromancer I had fettered my soul in the living world and the realms beyond. I had prepared for my return and was confident in my power.", npc, player)
+ elseif MsgContains(message, "confident") then
+ npcHandler:say("My soul wandered the plains of Zarganash, waiting for my wards to power up. Waiting for my soul to be slowly pulled back and manifest in the world of the living once again. What I had not taken into consideration was peace.", npc, player)
+ elseif MsgContains(message, "peace") then
+ npcHandler:say({
+ "Zarganash was not a place without its dangers, but for a soul as powerful as mine, there was little threat at all. For the first time in my existence I had to stop running forward. I had to wait for things to fit into their places. ...",
+ "And me, who had seen things that horrible, they would have obliterated a lesser man's mind, finally took the time to look back. And what I saw was frightening in its own right. ...",
+ "A great tiredness overcame me. With the flames of my ambitions calming down for the first time since I could remember, all my aspirations and plans seemed to petty and futile. ...",
+ "Everything I had worked for and my plans for the things to come seemed pointless, and the things I had lost and never allowed myself to experience weighed heavily on my soul.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "soul") then
+ npcHandler:say({
+ "I talked to other souls, lost in Zarganash, and most of them seemed like mirrors to myself. Their faults, their shortcomings, the things that were important to them and the things they had lost. ...",
+ "It was all like miniature copies of my own grand plans and losses. It made me think. And the great tiredness weighed even more heavy on me. A weariness of the world, of the hunger that drove me.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "weariness") then
+ npcHandler:say({
+ "Then I met a wise soul. A teacher that did not lecture. I never was impressed by anything but my own accomplishments. But the inner balance and peace of this soul, it did impress me. A lot. ...",
+ "I, who fancied myself to have been the epitome of knowledge, learned things that were entirely new to me. But this knowledge wasn't about power. It was about me.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "knowledge") then
+ npcHandler:say("I recognized the extent of my folly and failure. I decided not to return to the world of the living.", npc, player)
+ elseif MsgContains(message, "return") then
+ npcHandler:say({
+ "I decided to stay here, even pass on into the great beyond at some point. Yet I still feel the pull of my fetters. I can faintly hear those who think they are my followers, calling to me.",
+ "And I feel others, many others who crave my powers and try to bring me back for their own gain.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "fetters") then
+ npcHandler:say({
+ "Over my time in Zarganash I split away the parts of me that my worldly fetters were bound to. Yet I had to recognize that they are still a part of me and I'm bound to them. ...",
+ "The fetters and the efforts to call me back are empowering them. I feel them growing in strength and gaining awareness on their own. ...",
+ "They are beginning to feed not only on the fetters and incarnations but also on me. As I grow weaker, they grow more powerful over time.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "powerful") then
+ npcHandler:say("The only way I can get rid of them is to disperse them, to 'kill' them so to say. But they are tainted parts of myself and even going near them might endanger my sanity and stability. So all I can do is to ask you to do this daunting task.", npc, player)
+ elseif MsgContains(message, "task") then
+ local soulWarQuest = player:soulWarQuestKV()
+ -- Checks if the boss has already been defeated
+ if soulWarQuest:get("goshnar's-megalomania-killed") then
+ npcHandler:say({
+ "You did it! For the first time I can feel free from the pull of my past. Now I'm free at last. ...",
+ "I might stay a while and teach other souls about the inner peace, but will eventually pass on. Thank you so much, my hero. My eternal gratitude and blessings will be with you!",
+ }, npc, player, 2000)
+ npcHandler:setTopic(playerId, 2)
+ player:addOutfit("Revenant")
+ else
+ npcHandler:say("I'm aware I have no right to ask and I have little to offer as a payment, but I ask you nonetheless. Will you face my fettered vices and destroy them for me?", npc, player)
+ npcHandler:setTopic(playerId, 1)
+ end
+ elseif MsgContains(message, "yes") and npcHandler:getTopic(playerId) == 1 then
+ npcHandler:say("Thank you for accepting this burden.", npc, player)
+ soulWarQuest:set("teleport-access", true)
+ elseif MsgContains(message, "burden") then
+ npcHandler:say({
+ "You will have to reach each of the negative parts of my personality that I split away. They are hidden deep in the depths of Zarganash and will have corrupted and twisted their surroundings into dangerous nightmares. ...",
+ "Even worse, you'll likely encounter minions of those who want to claim my soul as their prize for their own depraved reasons. You will have to destroy my shards to set me free.",
+ }, npc, player, 5000)
+ elseif MsgContains(message, "shards") then
+ npcHandler:say("You haven't killed Malice yet. You haven't killed Hatred yet. You haven't killed Spite yet. You haven't killed Cruelty yet. You haven't killed Greed yet.", npc, player)
+ elseif MsgContains(message, "hate") then
+ npcHandler:say({
+ "I hated the world for its flaws and the reluctance of people to comply with my will. I was convinced I was destined for greatness and to change everything. Ordinary beings were far beneath me and my consideration. ...",
+ "All this opposition, all the wars were a nuisance on my way to greatness. I would have sacrificed the whole world to reach my goals.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "fermuba") then
+ npcHandler:say("My daughter was as ambitious as me, yet she lacked my intellect and my raw talent. She still was great and talented yet I sadly let her feel my disdain. One of the many errors that my way of hubris made me do.", npc, player)
+ elseif MsgContains(message, "ferumbras") then
+ npcHandler:say({
+ "Even in the lands of the dead, this one caused a stir. The dead were whispering his name. It made me feel jealous and angry at first, but at some point, after much self-reflection, I could recognize my own faults in the stories about him.",
+ "It was almost like looking into a mirror for the first time. However, he lived way later than me, and I never met his soul here, so I can't tell more about him.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "grandson") then
+ npcHandler:say(" I'm not aware of the fate of my linage. Neither I'm able to relate to the mortal world in that way. Each of us is an individual, not bound by ties of blood or herritage.", npc, player)
+ elseif MsgContains(message, "pale worm") then
+ npcHandler:say("His avatar might be destroyed for now and it'd grip on Zarganash considerably weakened. Yet he burrowed to deep into the essence of this realm to be annihilated this easy.", npc, player)
+ elseif MsgContains(message, "necromant king") then
+ npcHandler:say({
+ "They called me the necromant king, in an act of reverence, but to me it was always more of a slander. To limit my greatness to this insignificant aspect was an insult to my ego. But I let it slip for the greater good. ...",
+ "I felt it was beneath me to correct them and I went along.",
+ }, npc, player, 4000)
+ elseif MsgContains(message, "minions") or MsgContains(message, "followers") then
+ npcHandler:say("I despised my followers for their petty agendas and for their limited vision of my own goals and personality.", npc, player)
+ elseif MsgContains(message, "shards") then
+ local bossesYetToDefeat = {}
+ for bossName, _ in pairs(SoulWarQuest.miniBosses) do
+ if not soulWarQuest:get(bossName) then
+ table.insert(bossesYetToDefeat, bossName)
+ end
+ end
+
+ local message
+ if #bossesYetToDefeat > 0 then
+ message = "You haven't killed " .. table.concat(bossesYetToDefeat, ", ") .. " yet."
+ else
+ message = "You have defeated all the Goshnar's Bosses. Your soul shines brighter with each victory."
+ end
+ npcHandler:say(message, npc, player)
+ elseif MsgContains(message, "taints") or MsgContains(message, "penalties") then
+ if player:getTaintLevel() ~= nil then
+ player:resetTaints(true)
+ npcHandler:say("I have cleansed you from the taints that you carried with you. You are now free from the burden that you should not have to bear.", npc, player)
+ return
+ end
+
+ npcHandler:say("You are not tainted by the darkness of the world. You are pure and free from the burdens that others carry.", npc, player)
+ end
+ return true
+end
+
+npcHandler:setCallback(CALLBACK_MESSAGE_DEFAULT, playerSayCallback)
+
+npcHandler:setMessage(MESSAGE_GREET, "Be greeted, living soul!")
+
+npcHandler:addModule(FocusModule:new(), npcConfig.name, true, true, true)
+
+-- npcType registering the npcConfig table
+npcType:register(npcConfig)
diff --git a/data-otservbr-global/npc/maria.lua b/data-otservbr-global/npc/maria.lua
index 74780e4a85a..7c63e11ac02 100644
--- a/data-otservbr-global/npc/maria.lua
+++ b/data-otservbr-global/npc/maria.lua
@@ -65,8 +65,6 @@ npcConfig.shop = {
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
{ itemName = "tomato", clientId = 3596, buy = 5 },
{ itemName = "valentine's cake", clientId = 6392, buy = 100 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/mirabell.lua b/data-otservbr-global/npc/mirabell.lua
index 06362976254..db5f256ccd4 100644
--- a/data-otservbr-global/npc/mirabell.lua
+++ b/data-otservbr-global/npc/mirabell.lua
@@ -134,9 +134,6 @@ npcConfig.shop = {
{ itemName = "mug of lemonade", clientId = 2880, buy = 2, count = 12 },
{ itemName = "mug of water", clientId = 2880, buy = 1, count = 1 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 1, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/miraia.lua b/data-otservbr-global/npc/miraia.lua
index ad74b08040e..d8fbf586f2a 100644
--- a/data-otservbr-global/npc/miraia.lua
+++ b/data-otservbr-global/npc/miraia.lua
@@ -200,8 +200,6 @@ npcConfig.shop = {
{ itemName = "mug of milk", clientId = 2880, buy = 5, count = 9 },
{ itemName = "mug of water", clientId = 2880, buy = 2, count = 1 },
{ itemName = "scarab cheese", clientId = 169, buy = 100 },
- { itemName = "vial of milk", clientId = 2874, buy = 5, count = 1, subType = 9 },
- { itemName = "vial of water", clientId = 2874, buy = 2, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/norma.lua b/data-otservbr-global/npc/norma.lua
index 769bf7b7b3f..693dc7fadbb 100644
--- a/data-otservbr-global/npc/norma.lua
+++ b/data-otservbr-global/npc/norma.lua
@@ -200,9 +200,6 @@ npcConfig.shop = {
{ itemName = "mug of milk", clientId = 2880, buy = 2, count = 6 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
{ itemName = "party cake", clientId = 6279, buy = 50 },
- { itemName = "vial of beer", clientId = 2874, buy = 3, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of milk", clientId = 2874, buy = 2, count = 1, subType = 9 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/pugwah.lua b/data-otservbr-global/npc/pugwah.lua
index e9c0874d5aa..9e9007a7667 100644
--- a/data-otservbr-global/npc/pugwah.lua
+++ b/data-otservbr-global/npc/pugwah.lua
@@ -61,9 +61,6 @@ npcConfig.shop = {
{ itemName = "mug of lemonade", clientId = 2880, buy = 2, count = 12 },
{ itemName = "mug of milk", clientId = 2880, buy = 2, count = 6 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
- { itemName = "vial of beer", clientId = 2874, buy = 3, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 4, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 2, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/ramina.lua b/data-otservbr-global/npc/ramina.lua
index d88b305aaeb..f9b253c477e 100644
--- a/data-otservbr-global/npc/ramina.lua
+++ b/data-otservbr-global/npc/ramina.lua
@@ -64,9 +64,7 @@ npcConfig.shop = {
{ itemName = "mug of wine", clientId = 2880, buy = 15, count = 2 },
{ itemName = "orange", clientId = 3586, buy = 12 },
{ itemName = "peas", clientId = 11683, buy = 5 },
- { itemName = "vial of fruit juice", clientId = 2874, buy = 10, count = 14 },
- { itemName = "vial of water", clientId = 2874, buy = 2, count = 1 },
- { itemName = "vial of wine", clientId = 2874, buy = 15, count = 1, subType = 2 },
+ { itemName = "vial of water", clientId = 2874, buy = 10, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/red_lilly.lua b/data-otservbr-global/npc/red_lilly.lua
index 357cb02775b..b744fae905d 100644
--- a/data-otservbr-global/npc/red_lilly.lua
+++ b/data-otservbr-global/npc/red_lilly.lua
@@ -85,7 +85,7 @@ npcConfig.shop = {
{ itemName = "torch", clientId = 2920, buy = 2 },
{ itemName = "vial", clientId = 2874, sell = 5 },
{ itemName = "vial of oil", clientId = 2874, buy = 20, count = 7 },
- { itemName = "vial of water", clientId = 2874, buy = 10, count = 1 },
+ { itemName = "vial of water", clientId = 2874, buy = 20, count = 1 },
{ itemName = "watch", clientId = 2906, buy = 20, sell = 6 },
{ itemName = "waterskin of water", clientId = 2901, buy = 10, count = 1 },
{ itemName = "wooden hammer", clientId = 3459, sell = 15 },
diff --git a/data-otservbr-global/npc/swolt.lua b/data-otservbr-global/npc/swolt.lua
index 7caa8aa4160..5d8a7b48830 100644
--- a/data-otservbr-global/npc/swolt.lua
+++ b/data-otservbr-global/npc/swolt.lua
@@ -86,8 +86,6 @@ npcConfig.shop = {
{ itemName = "mug of beer", clientId = 2880, buy = 2, count = 3 },
{ itemName = "mug of water", clientId = 2880, buy = 1, count = 1 },
{ itemName = "terramite eggs", clientId = 10453, sell = 50 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of water", clientId = 2874, buy = 1, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/tanaro.lua b/data-otservbr-global/npc/tanaro.lua
index 467b70989ae..bf43db15d79 100644
--- a/data-otservbr-global/npc/tanaro.lua
+++ b/data-otservbr-global/npc/tanaro.lua
@@ -67,8 +67,6 @@ npcConfig.shop = {
{ itemName = "salmon", clientId = 3579, buy = 6 },
{ itemName = "valentine's cake", clientId = 6392, buy = 100 },
{ itemName = "white mushroom", clientId = 3723, buy = 6 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 2, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/npc/urkalio.lua b/data-otservbr-global/npc/urkalio.lua
index 5919a42d761..dbcbbc238b1 100644
--- a/data-otservbr-global/npc/urkalio.lua
+++ b/data-otservbr-global/npc/urkalio.lua
@@ -62,9 +62,6 @@ npcConfig.shop = {
{ itemName = "mug of lemonade", clientId = 2880, buy = 2, count = 12 },
{ itemName = "mug of water", clientId = 2880, buy = 1, count = 1 },
{ itemName = "mug of wine", clientId = 2880, buy = 3, count = 2 },
- { itemName = "vial of beer", clientId = 2874, buy = 2, count = 1, subType = 3 },
- { itemName = "vial of wine", clientId = 2874, buy = 3, count = 1, subType = 2 },
- { itemName = "vial of water", clientId = 2874, buy = 1, count = 1 },
}
-- On buy npc shop message
npcType.onBuyItem = function(npc, player, itemId, subType, amount, ignore, inBackpacks, totalCost)
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_cruelty.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_cruelty.lua
deleted file mode 100644
index 2e01880589d..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_cruelty.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Cruelty",
- position = Position(33856, 31866, 7),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33854, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
- { pos = Position(33855, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
- { pos = Position(33856, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
- { pos = Position(33857, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
- { pos = Position(33858, 31854, 6), teleport = Position(33856, 31872, 7), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33847, 31858, 7),
- to = Position(33864, 31874, 7),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33853, y = 31854, z = 6 })
-lever:register()
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_greed.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_greed.lua
deleted file mode 100644
index 522cde76845..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_greed.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Greed",
- position = Position(33746, 31666, 14),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33776, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33777, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33778, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33779, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33780, 31665, 14), teleport = Position(33747, 31671, 14), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33737, 31658, 14),
- to = Position(33755, 31673, 14),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33775, y = 31665, z = 14 })
-lever:register()
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_hatred.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_hatred.lua
deleted file mode 100644
index 963404c8d85..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_hatred.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Hatred",
- position = Position(33744, 31599, 14),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33773, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33774, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33775, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33776, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33777, 31601, 14), teleport = Position(33743, 31604, 14), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33735, 31592, 14),
- to = Position(33751, 31606, 14),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33772, y = 31601, z = 14 })
-lever:register()
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_malice.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_malice.lua
deleted file mode 100644
index 44d102ad598..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_malice.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Malice",
- position = Position(33710, 31599, 14),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33679, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33680, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33681, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33682, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33683, 31599, 14), teleport = Position(33710, 31605, 14), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33699, 31590, 14),
- to = Position(33718, 31607, 14),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33678, y = 31599, z = 14 })
-lever:register()
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_megalomania.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_megalomania.lua
deleted file mode 100644
index 5896faa74c4..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_megalomania.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Megalomania",
- position = Position(33710, 31634, 14),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33676, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33677, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33678, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33679, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33680, 31634, 14), teleport = Position(33710, 31639, 14), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33701, 31626, 14),
- to = Position(33719, 31642, 14),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33675, y = 31634, z = 14 })
-lever:register()
diff --git a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_spite.lua b/data-otservbr-global/scripts/actions/bosses_levers/goshnar_spite.lua
deleted file mode 100644
index 0fd2396ae03..00000000000
--- a/data-otservbr-global/scripts/actions/bosses_levers/goshnar_spite.lua
+++ /dev/null
@@ -1,23 +0,0 @@
-local config = {
- boss = {
- name = "Goshnar's Spite",
- position = Position(33743, 31632, 14),
- },
- requiredLevel = 250,
- playerPositions = {
- { pos = Position(33774, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33775, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33776, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33777, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
- { pos = Position(33778, 31634, 14), teleport = Position(33742, 31639, 14), effect = CONST_ME_TELEPORT },
- },
- specPos = {
- from = Position(33734, 31624, 14),
- to = Position(33751, 31640, 14),
- },
- exit = Position(33621, 31427, 10),
-}
-
-local lever = BossLever(config)
-lever:position({ x = 33773, y = 31634, z = 14 })
-lever:register()
diff --git a/data-otservbr-global/scripts/lib/register_actions.lua b/data-otservbr-global/scripts/lib/register_actions.lua
index 6df6b96473a..6118f3b837a 100644
--- a/data-otservbr-global/scripts/lib/register_actions.lua
+++ b/data-otservbr-global/scripts/lib/register_actions.lua
@@ -1,106 +1,12 @@
-local holeId = {
- 294,
- 369,
- 370,
- 385,
- 394,
- 411,
- 412,
- 413,
- 432,
- 433,
- 435,
- 8709,
- 594,
- 595,
- 615,
- 609,
- 610,
- 615,
- 1156,
- 482,
- 483,
- 868,
- 874,
- 4824,
- 7768,
- 433,
- 432,
- 413,
- 7767,
- 411,
- 370,
- 369,
- 7737,
- 7755,
- 7768,
- 7767,
- 7515,
- 7516,
- 7517,
- 7518,
- 7519,
- 7520,
- 7521,
- 7522,
- 7762,
- 8144,
- 8690,
- 8709,
- 12203,
- 12961,
- 17239,
- 19220,
- 23364,
-}
-
-local Itemsgrinder = {
+local holeId = { 294, 369, 370, 385, 394, 411, 412, 413, 432, 433, 435, 482, 483, 594, 595, 609, 610, 615, 868, 874, 1156, 4824, 7515, 7516, 7517, 7518, 7519, 7520, 7521, 7522, 7737, 7755, 7762, 7767, 7768, 8144, 8690, 8709, 12203, 12961, 17239, 19220, 23364 }
+local itemsGrinder = {
[675] = { item_id = 30004, effect = CONST_ME_BLUE_FIREWORKS }, -- Sapphire dust
[16122] = { item_id = 21507, effect = CONST_ME_GREENSMOKE }, -- Pinch of crystal dust
}
-
-local holes = {
- 593,
- 606,
- 608,
- 867,
- 21341,
-}
-
-local JUNGLE_GRASS = {
- 3696,
- 3702,
- 17153,
-}
-local WILD_GROWTH = {
- 2130,
- 2130,
- 2982,
- 2524,
- 2030,
- 2029,
- 10182,
-}
-
-local fruits = {
- 3584,
- 3585,
- 3586,
- 3587,
- 3588,
- 3589,
- 3590,
- 3591,
- 3592,
- 3593,
- 3595,
- 3596,
- 5096,
- 8011,
- 8012,
- 8013,
-}
-
+local holes = { 593, 606, 608, 867, 21341 }
+local jungleGrass = { 3696, 3702, 17153 }
+local wildGrowth = { 2130, 3635, 30224 }
+local fruits = { 3584, 3585, 3586, 3587, 3588, 3589, 3590, 3591, 3592, 3593, 3595, 3596, 5096, 8011, 8012, 8013 }
local lava = {
Position(32808, 32336, 11),
Position(32809, 32336, 11),
@@ -867,13 +773,13 @@ function onUsePick(player, item, fromPosition, target, toPosition, isHotkey)
end
function onUseMachete(player, item, fromPosition, target, toPosition, isHotkey)
- if table.contains(JUNGLE_GRASS, target.itemid) then
+ if table.contains(jungleGrass, target.itemid) then
target:transform(target.itemid == 17153 and 17151 or target.itemid - 1)
target:decay()
return true
end
- if table.contains(WILD_GROWTH, target.itemid) then
+ if table.contains(wildGrowth, target.itemid) then
toPosition:sendMagicEffect(CONST_ME_POFF)
target:remove()
return true
@@ -1035,14 +941,17 @@ function onUseScythe(player, item, fromPosition, target, toPosition, isHotkey)
target:transform(5463)
target:decay()
Game.createItem(5466, 1, toPosition)
+ return true
elseif target.itemid == 3653 then -- wheat
target:transform(3651)
target:decay()
Game.createItem(3605, 1, toPosition)
+ return true
elseif target.itemid == 30623 then -- reed
target:transform(30624)
target:decay()
Game.createItem(30975, 1, toPosition)
+ return true
-- The secret library
elseif toPosition == Position(32177, 31925, 7) then
player:teleportTo({ x = 32515, y = 32535, z = 12 })
@@ -1135,7 +1044,7 @@ function onGrindItem(player, item, fromPosition, target, toPosition)
if not (target.itemid == 21573) then
return false
end
- for index, value in pairs(Itemsgrinder) do
+ for index, value in pairs(itemsGrinder) do
if item.itemid == index then
local topParent = item:getTopParent()
if topParent.isItem and (not topParent:isItem() or topParent.itemid ~= 470) then
diff --git a/data-otservbr-global/scripts/movements/teleport/dragolisk_teleport.lua b/data-otservbr-global/scripts/movements/teleport/dragolisk_teleport.lua
new file mode 100644
index 00000000000..2c04c3f2a39
--- /dev/null
+++ b/data-otservbr-global/scripts/movements/teleport/dragolisk_teleport.lua
@@ -0,0 +1,26 @@
+local function teleportToDestination(creature, position)
+ if not creature:isPlayer() then
+ return true
+ end
+
+ if position.x == 33216 and position.y == 31126 and position.z == 14 then
+ creature:teleportTo(Position(33186, 31190, 7))
+ creature:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+ elseif position.x == 33187 and position.y == 31190 and position.z == 7 then
+ creature:teleportTo(Position(33217, 31123, 14))
+ creature:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+ else
+ return false
+ end
+end
+
+local dragoliskTeleport = MoveEvent()
+function dragoliskTeleport.onStepIn(creature, item, position, fromPosition)
+ return teleportToDestination(creature, position)
+end
+
+dragoliskTeleport:position({ x = 33216, y = 31126, z = 14 })
+dragoliskTeleport:position({ x = 33187, y = 31190, z = 7 })
+dragoliskTeleport:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/action-reward_soul_war.lua b/data-otservbr-global/scripts/quests/soul_war/action-reward_soul_war.lua
new file mode 100644
index 00000000000..fae3fc59794
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/action-reward_soul_war.lua
@@ -0,0 +1,60 @@
+local rewardSoulWar = Action()
+
+function rewardSoulWar.onUse(creature, item, fromPosition, target, toPosition, isHotkey)
+ local rewardItem = SoulWarQuest.finalRewards[math.random(1, #SoulWarQuest.finalRewards)]
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ if soulWarQuest:get("final-reward") then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have already received your reward.")
+ return true
+ end
+
+ if not soulWarQuest:get("goshnar's-megalomania-killed") then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need to defeat Goshnar's Megalomania to receive your reward.")
+ return true
+ end
+
+ player:addItem(rewardItem.id, 1)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have found a " .. rewardItem.name .. ".")
+ soulWarQuest:set("final-reward", true)
+ return true
+end
+
+rewardSoulWar:position({ x = 33620, y = 31400, z = 10 })
+rewardSoulWar:register()
+
+local phantasmalJadeMount = Action()
+
+function phantasmalJadeMount.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ local soulWarQuest = player:soulWarQuestKV()
+ if soulWarQuest:get("panthasmal-jade-mount") then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You already have Phantasmal Jade mount!")
+ return true
+ end
+
+ if table.contains({ 34072, 34073, 34074 }, item.itemid) then
+ if player:getItemCount(34072) >= 4 and player:getItemCount(34073) == 1 and player:getItemCount(34074) == 1 then
+ player:removeItem(34072, 4)
+ player:removeItem(34073, 1)
+ player:removeItem(34074, 1)
+ player:addMount(167)
+ player:addAchievement("You got Horse Power")
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Congratulations! You won Phantasmal Jade mount.")
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Congratulations! You won You got Horse Power achievement.")
+ player:getPosition():sendMagicEffect(CONST_ME_HOLYDAMAGE)
+ soulWarQuest:set("panthasmal-jade-mount", true)
+ else
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You don't have the necessary items!")
+ player:getPosition():sendMagicEffect(CONST_ME_POFF)
+ end
+ end
+
+ return true
+end
+
+phantasmalJadeMount:id(34072, 34073, 34074)
+phantasmalJadeMount:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/actions_bosses_killed.lua b/data-otservbr-global/scripts/quests/soul_war/actions_bosses_killed.lua
deleted file mode 100644
index de4b8ae182f..00000000000
--- a/data-otservbr-global/scripts/quests/soul_war/actions_bosses_killed.lua
+++ /dev/null
@@ -1,24 +0,0 @@
-local bosses = {
- ["goshnar's malice"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarMaliceKilled },
- ["goshnar's hatred"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarHatredKilled },
- ["goshnar's spite"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarSpiteKilled },
- ["goshnar's cruelty"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarCrueltyKilled },
- ["goshnar's greed"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarGreedKilled },
- ["goshnar's megalomania"] = { storage = Storage.Quest.U12_40.SoulWar.GoshnarMegalomaniaKilled },
-}
-
-local bossesSoulWar = CreatureEvent("SoulwarsBossDeath")
-function bossesSoulWar.onDeath(creature)
- local bossConfig = bosses[creature:getName():lower()]
- if not bossConfig then
- return true
- end
- onDeathForDamagingPlayers(creature, function(creature, player)
- if bossConfig.storage then
- player:setStorageValue(bossConfig.storage, 1)
- end
- end)
- return true
-end
-
-bossesSoulWar:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/actions_portal_megalomania.lua b/data-otservbr-global/scripts/quests/soul_war/actions_portal_megalomania.lua
deleted file mode 100644
index 41d3763c1c7..00000000000
--- a/data-otservbr-global/scripts/quests/soul_war/actions_portal_megalomania.lua
+++ /dev/null
@@ -1,38 +0,0 @@
-local storagesTable = {
- { storage = Storage.Quest.U12_40.SoulWar.GoshnarMaliceKilled, bossName = "Goshnar's Malice" },
- { storage = Storage.Quest.U12_40.SoulWar.GoshnarHatredKilled, bossName = "Goshnar's Hatred" },
- { storage = Storage.Quest.U12_40.SoulWar.GoshnarSpiteKilled, bossName = "Goshnar's Spite" },
- { storage = Storage.Quest.U12_40.SoulWar.GoshnarCrueltyKilled, bossName = "Goshnar's Cruelty" },
- { storage = Storage.Quest.U12_40.SoulWar.GoshnarGreedKilled, bossName = "Goshnar's Greed" },
-}
-
-local portalMegalomania = MoveEvent()
-function portalMegalomania.onStepIn(creature, item, position, fromPosition)
- local player = creature:getPlayer()
- if not player then
- return false
- end
- if player:getLevel() < 250 then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need at least level 250 to enter.")
- player:teleportTo(fromPosition, true)
- player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
- return false
- end
- local text = ""
- for value in pairs(storagesTable) do
- if player:getStorageValue(storagesTable[value].storage) < 0 then
- text = text .. "\n" .. storagesTable[value].bossName
- end
- end
- if text == "" then
- return true
- else
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You still need to defeat:" .. text)
- player:teleportTo(fromPosition, true)
- return false
- end
-end
-
-portalMegalomania:type("stepin")
-portalMegalomania:position({ x = 33611, y = 31430, z = 10 })
-portalMegalomania:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/actions_reward_soul_war.lua b/data-otservbr-global/scripts/quests/soul_war/actions_reward_soul_war.lua
deleted file mode 100644
index 1e3c9c773e0..00000000000
--- a/data-otservbr-global/scripts/quests/soul_war/actions_reward_soul_war.lua
+++ /dev/null
@@ -1,89 +0,0 @@
-local rewards = {
- { id = 34082, name = "Soulcutter" },
- { id = 34083, name = "Soulshredder" },
- { id = 34084, name = "Soulbiter" },
- { id = 34085, name = "Souleater" },
- { id = 34086, name = "Soulcrusher" },
- { id = 34087, name = "Soulmaimer" },
- { id = 34088, name = "Soulbleeder" },
- { id = 34089, name = "Soulpiercer" },
- { id = 34090, name = "Soultainter" },
- { id = 34091, name = "Soulhexer" },
- { id = 34092, name = "Soulshanks" },
- { id = 34093, name = "Soulstrider" },
- { id = 34094, name = "Soulshell" },
- { id = 34095, name = "Soulmantel" },
- { id = 34096, name = "Soulshroud" },
- { id = 34097, name = "Pair of Soulwalkers" },
- { id = 34098, name = "Pair of Soulstalkers" },
- { id = 34099, name = "Soulbastion" },
-}
-local outfits = { 1322, 1323 }
-
-local function addOutfits(player)
- if player:getStorageValue(Storage.Quest.U12_40.SoulWar.OutfitReward) < 0 then
- player:addOutfit(outfits[1], 0)
- player:addOutfit(outfits[2], 0)
- player:setStorageValue(Storage.Quest.U12_40.SoulWar.OutfitReward, 1)
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Congratulations you received the Revenant Outfit.")
- end
-end
-
-local rewardSoulWar = Action()
-function rewardSoulWar.onUse(creature, item, fromPosition, target, toPosition, isHotkey)
- local randId = math.random(1, #rewards)
- local rewardItem = rewards[randId]
- local player = creature:getPlayer()
- if not player then
- return false
- end
- if player:getStorageValue(Storage.Quest.U12_40.SoulWar.QuestReward) < 0 then
- player:addItem(rewardItem.id, 1)
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have found a " .. rewardItem.name .. ".")
- player:setStorageValue(Storage.Quest.U12_40.SoulWar.QuestReward, 1)
- addOutfits(player)
- return true
- else
- addOutfits(player)
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have already collected your reward")
- return false
- end
-end
-
-rewardSoulWar:position({ x = 33620, y = 31400, z = 10 })
-rewardSoulWar:register()
-
------------------------------
--- Phantasmal Jade Mount function
-
-local phantasmalJadeMount = Action()
-function phantasmalJadeMount.onUse(player, item, fromPosition, target, toPosition, isHotkey)
- local storage = Storage.Quest.U12_40.SoulWar.MountReward
- if player:getStorageValue(storage) == 1 then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You already have Phantasmal Jade mount!")
- return false
- end
-
- if table.contains({ 34072, 34073, 34074 }, item.itemid) then
- -- check items
- if player:getItemCount(34072) >= 4 and player:getItemCount(34073) == 1 and player:getItemCount(34074) == 1 then
- player:removeItem(34072, 4)
- player:removeItem(34073, 1)
- player:removeItem(34074, 1)
- player:addMount(167)
- player:setStorageValue(storage, 1)
- player:addAchievement("You got Horse Power")
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Congratulations! You won Phantasmal Jade mount.")
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Congratulations! You won You got Horse Power achievement.")
- player:getPosition():sendMagicEffect(CONST_ME_HOLYDAMAGE)
- return true
- else
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You don't have the necessary items!")
- player:getPosition():sendMagicEffect(CONST_ME_POFF)
- return false
- end
- end
-end
-
-phantasmalJadeMount:id(34072, 34073, 34074)
-phantasmalJadeMount:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/actions_soulwar_entrances.lua b/data-otservbr-global/scripts/quests/soul_war/actions_soulwar_entrances.lua
deleted file mode 100644
index 51f8281a04d..00000000000
--- a/data-otservbr-global/scripts/quests/soul_war/actions_soulwar_entrances.lua
+++ /dev/null
@@ -1,63 +0,0 @@
-local config = {
- { position = { x = 33615, y = 31422, z = 10 }, destination = { x = 34009, y = 31014, z = 9 } }, -- hunt infernal demon
- { position = { x = 33618, y = 31422, z = 10 }, destination = { x = 33972, y = 31041, z = 11 } }, -- hunt rotten
- { position = { x = 33621, y = 31422, z = 10 }, destination = { x = 33894, y = 31019, z = 8 } }, -- hunt bony sea devil
- { position = { x = 33624, y = 31422, z = 10 }, destination = { x = 33858, y = 31831, z = 3 } }, -- hunt cloak
- { position = { x = 33627, y = 31422, z = 10 }, destination = { x = 33887, y = 31188, z = 10 } }, -- hunt many faces
- { position = { x = 33950, y = 31109, z = 8 }, destination = { x = 33780, y = 31634, z = 14 } }, -- goshnar's spite entrance
- { position = { x = 33937, y = 31217, z = 11 }, destination = { x = 33782, y = 31665, z = 14 } }, -- goshnar's greed entrance
- { position = { x = 34022, y = 31091, z = 11 }, destination = { x = 33685, y = 31599, z = 14 } }, -- goshnar's malice entrance
- { position = { x = 33856, y = 31884, z = 5 }, destination = { x = 33857, y = 31865, z = 6 } }, -- goshnar's cruelty entrance
- { position = { x = 33889, y = 31873, z = 3 }, destination = { x = 33830, y = 31881, z = 4 } }, -- 1st to 2nd floor cloak
- { position = { x = 33829, y = 31880, z = 4 }, destination = { x = 33856, y = 31889, z = 5 } }, -- 2nd to 3rd floor cloak
-}
-
-local portal = { position = { x = 33914, y = 31032, z = 12 }, destination = { x = 33780, y = 31601, z = 14 } } -- goshnar's hatred entrance
-
-local soulWarEntrances = MoveEvent()
-function soulWarEntrances.onStepIn(creature, item, position, fromPosition)
- local player = creature:getPlayer()
- if not player then
- return false
- end
- if player:getLevel() < 250 then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need at least level 250 to enter.")
- player:teleportTo(fromPosition, true)
- player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
- return false
- end
- for value in pairs(config) do
- if Position(config[value].position) == player:getPosition() then
- player:teleportTo(Position(config[value].destination))
- player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
- return true
- end
- end
-end
-
-soulWarEntrances:type("stepin")
-for value in pairs(config) do
- soulWarEntrances:position(config[value].position)
-end
-soulWarEntrances:register()
-
-local portalHatred = Action()
-function portalHatred.onUse(creature, item, position, fromPosition)
- local player = creature:getPlayer()
- if not player then
- return false
- end
- if player:getLevel() < 250 then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need at least level 250 to enter.")
- player:teleportTo(fromPosition, true)
- player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
- return false
- end
- doSendMagicEffect(item:getPosition(), CONST_ME_TELEPORT)
- player:teleportTo(Position(portal.destination))
- player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
- return true
-end
-
-portalHatred:position(portal.position)
-portalHatred:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/eventcallback_on_combat_taint.lua b/data-otservbr-global/scripts/quests/soul_war/eventcallback_on_combat_taint.lua
new file mode 100644
index 00000000000..3e5ec34f930
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/eventcallback_on_combat_taint.lua
@@ -0,0 +1,126 @@
+local taintCooldown = {}
+
+local function createTeleportEffect(position)
+ position:sendMagicEffect(CONST_ME_TELEPORT)
+end
+
+local function scheduleMonsterCreation(player, monster, monsterName, spawnPosition)
+ addEvent(createTeleportEffect, 1000, spawnPosition)
+ addEvent(createTeleportEffect, 2000, spawnPosition)
+ addEvent(createTeleportEffect, 3000, spawnPosition)
+
+ addEvent(function(playerId, monsterId)
+ local eventPlayer = Player(playerId)
+ if not eventPlayer then
+ return
+ end
+
+ local eventMonster = Monster(monsterId)
+ if not eventMonster or eventMonster:isDead() then
+ return
+ end
+
+ -- Only create if the player not have cooldown
+ if not taintCooldown[playerId] or os.time() > taintCooldown[playerId] then
+ taintCooldown[playerId] = os.time() + 30
+ local monster = Game.createMonster(monsterName, spawnPosition, true, true)
+ if monster then
+ spawnPosition:sendMagicEffect(CONST_ME_TELEPORT)
+ logger.debug("Spamming monster with name {} to player {}", monsterName, eventPlayer:getName())
+ end
+ end
+ end, 4000, player:getId(), monster:getId())
+end
+
+local function onPlayerAttackMonster(player, target)
+ local monster = target:getMonster()
+ if not monster then
+ return
+ end
+
+ -- It will only execute if the player has the second taint
+ if player:getTaintNameByNumber(2) ~= nil then
+ local chance = math.random(1, 200)
+ local spawnPosition = player:getPosition()
+ if chance == 1 then -- 0.5% chance
+ local foundMonsterName = player:getSoulWarZoneMonster()
+ if foundMonsterName ~= nil then
+ scheduleMonsterCreation(player, monster, foundMonsterName, spawnPosition)
+ end
+ end
+ end
+end
+
+local function onMonsterAttackPlayer(target, primaryValue, secondaryValue)
+ local targetPlayer = target:getPlayer()
+ if not targetPlayer then
+ return primaryValue, secondaryValue
+ end
+
+ if targetPlayer:getTaintNameByNumber(3) ~= nil then
+ local monsterZone = targetPlayer:getSoulWarZoneMonster()
+ if monsterZone ~= nil then
+ logger.debug("Player {} have third taint, primary value {}, secondary {}", targetPlayer:getName(), primaryValue, secondaryValue)
+ primaryValue = primaryValue + math.ceil(primaryValue * 0.15)
+ secondaryValue = secondaryValue + math.ceil(secondaryValue * 0.15)
+ logger.debug("Primary value after {}, secondary {}", primaryValue, secondaryValue)
+ end
+ end
+
+ return primaryValue, secondaryValue
+end
+
+local callback = EventCallback("CreatureOnCombatTaint")
+
+function callback.creatureOnCombat(caster, target, primaryValue, primaryType, secondaryValue, secondaryType, origin)
+ if not caster or not target then
+ return primaryValue, primaryType, secondaryValue, secondaryType
+ end
+
+ -- Second taint
+ local attackerPlayer = caster:getPlayer()
+ if attackerPlayer and target:isMonster() then
+ onPlayerAttackMonster(attackerPlayer, target)
+ end
+
+ -- Third taint
+ if caster:getMonster() then
+ primaryValue, secondaryValue = onMonsterAttackPlayer(target, primaryValue, secondaryValue)
+ end
+
+ return primaryValue, primaryType, secondaryValue, secondaryType
+end
+
+callback:register()
+
+callback = EventCallback("PlayerOnThinkTaint")
+
+local accumulatedTime = {}
+
+function callback.playerOnThink(player, interval)
+ if not player then
+ return
+ end
+
+ local playerId = player:getId()
+ if not accumulatedTime[playerId] then
+ accumulatedTime[playerId] = 0
+ end
+
+ accumulatedTime[playerId] = accumulatedTime[playerId] + interval
+
+ if accumulatedTime[playerId] >= 10000 then
+ local soulWarQuest = player:soulWarQuestKV()
+ if player:getSoulWarZoneMonster() ~= nil and player:getTaintNameByNumber(5) ~= nil then
+ local hpLoss = math.ceil(player:getHealth() * 0.1)
+ local manaLoss = math.ceil(player:getMana() * 0.1)
+ player:addHealth(-hpLoss)
+ player:addMana(-manaLoss)
+ logger.debug("Fifth taint removing '{}' mana and '{}' health from player {}", manaLoss, hpLoss, player:getName())
+ end
+
+ accumulatedTime[playerId] = 0
+ end
+end
+
+callback:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/globalevent-ebb_and_flow_change_maps.lua b/data-otservbr-global/scripts/quests/soul_war/globalevent-ebb_and_flow_change_maps.lua
new file mode 100644
index 00000000000..d13b5cf2cd6
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/globalevent-ebb_and_flow_change_maps.lua
@@ -0,0 +1,135 @@
+local function updateWaterPoolsSize()
+ for _, pos in ipairs(SoulWarQuest.ebbAndFlow.poolPositions) do
+ local tile = Tile(pos)
+ if tile then
+ local item = tile:getItemById(SoulWarQuest.ebbAndFlow.smallPoolId)
+ if item then
+ item:transform(SoulWarQuest.ebbAndFlow.MediumPoolId)
+ -- Starts another timer for filling after an additional 40 seconds
+ addEvent(function()
+ local item = tile:getItemById(SoulWarQuest.ebbAndFlow.MediumPoolId)
+ if item then
+ item:transform(SoulWarQuest.ebbAndFlow.smallPoolId)
+ end
+ end, 40000) -- 40 seconds
+ end
+ end
+ end
+end
+
+local function loadMapEmpty()
+ if SoulWarQuest.ebbAndFlow.getZone():countPlayers() > 0 then
+ local players = SoulWarQuest.ebbAndFlow.getZone():getPlayers()
+ for _, player in ipairs(players) do
+ if player:getPosition().z == 8 then
+ if player:isInBoatSpot() then
+ local teleportPosition = player:getPosition()
+ teleportPosition.z = 9
+ player:teleportTo(teleportPosition)
+ logger.trace("Teleporting player to down.")
+ end
+ player:sendCreatureAppear()
+ end
+ end
+ end
+
+ Game.loadMap(SoulWarQuest.ebbAndFlow.mapsPath.empty)
+ SoulWarQuest.ebbAndFlow.setLoadedEmptyMap(true)
+ SoulWarQuest.ebbAndFlow.setActive(false)
+
+ local updatePlayers = EventCallback("UpdatePlayersEmptyEbbFlowMap")
+ function updatePlayers.mapOnLoad(mapPath)
+ if mapPath ~= SoulWarQuest.ebbAndFlow.mapsPath.empty then
+ return
+ end
+
+ SoulWarQuest.ebbAndFlow.updateZonePlayers()
+ end
+
+ updatePlayers:register()
+
+ addEvent(function()
+ -- Change the appearance of puddles to indicate the next filling
+ updateWaterPoolsSize()
+ end, 80000) -- 80 seconds
+end
+
+local function getDistance(pos1, pos2)
+ return math.sqrt((pos1.x - pos2.x) ^ 2 + (pos1.y - pos2.y) ^ 2 + (pos1.z - pos2.z) ^ 2)
+end
+
+local function findNearestRoomPosition(playerPosition)
+ local nearestPosition = nil
+ local smallestDistance = nil
+ for _, room in ipairs(SoulWarQuest.ebbAndFlow.centerRoomPositions) do
+ local distance = getDistance(playerPosition, room.conor)
+ if not smallestDistance or distance < smallestDistance then
+ smallestDistance = distance
+ nearestPosition = room.teleportPosition
+ end
+ end
+ return nearestPosition
+end
+
+local function loadMapInundate()
+ if SoulWarQuest.ebbAndFlow.getZone():countPlayers() > 0 then
+ local players = SoulWarQuest.ebbAndFlow.getZone():getPlayers()
+ for _, player in ipairs(players) do
+ local playerPosition = player:getPosition()
+ if playerPosition.z == 9 then
+ if player:isInBoatSpot() then
+ local nearestCenterPosition = findNearestRoomPosition(playerPosition)
+ player:teleportTo(nearestCenterPosition)
+ logger.trace("Teleporting player to the near center position room and updating tile.")
+ else
+ player:teleportTo(SoulWarQuest.ebbAndFlow.waitPosition)
+ logger.trace("Teleporting player to wait position and updating tile.")
+ end
+ playerPosition:sendMagicEffect(CONST_ME_TELEPORT)
+ end
+ player:sendCreatureAppear()
+ end
+ end
+
+ Game.loadMap(SoulWarQuest.ebbAndFlow.mapsPath.inundate)
+ SoulWarQuest.ebbAndFlow.setLoadedEmptyMap(false)
+ SoulWarQuest.ebbAndFlow.setActive(true)
+
+ local updatePlayers = EventCallback("UpdatePlayersInundateEbbFlowMap")
+ function updatePlayers.mapOnLoad(mapPath)
+ if mapPath ~= SoulWarQuest.ebbAndFlow.mapsPath.inundate then
+ return
+ end
+
+ SoulWarQuest.ebbAndFlow.updateZonePlayers()
+ end
+
+ updatePlayers:register()
+end
+
+local loadEmptyMap = GlobalEvent("SoulWarQuest.ebbAndFlow")
+
+function loadEmptyMap.onStartup()
+ Game.loadMap(SoulWarQuest.ebbAndFlow.mapsPath.ebbFlow)
+ loadMapEmpty()
+ SoulWarQuest.ebbAndFlow.updateZonePlayers()
+end
+
+loadEmptyMap:register()
+
+local eddAndFlowInundate = GlobalEvent("eddAndFlowInundate")
+
+function eddAndFlowInundate.onThink(interval, lastExecution)
+ if SoulWarQuest.ebbAndFlow.isLoadedEmptyMap() then
+ logger.trace("Map change to empty in {} minutes.", SoulWarQuest.ebbAndFlow.intervalChangeMap)
+ loadMapInundate()
+ elseif SoulWarQuest.ebbAndFlow.isActive() then
+ logger.trace("Map change to inundate in {} minutes.", SoulWarQuest.ebbAndFlow.intervalChangeMap)
+ loadMapEmpty()
+ end
+
+ return true
+end
+
+eddAndFlowInundate:interval(SoulWarQuest.ebbAndFlow.intervalChangeMap * 60 * 1000)
+eddAndFlowInundate:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/moveevent-claustrophobic-inferno-raid.lua b/data-otservbr-global/scripts/quests/soul_war/moveevent-claustrophobic-inferno-raid.lua
new file mode 100644
index 00000000000..0502d53e365
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/moveevent-claustrophobic-inferno-raid.lua
@@ -0,0 +1,61 @@
+local firstRaid = MoveEvent()
+local secondRaid = MoveEvent()
+local thirdRaid = MoveEvent()
+
+local spawnMonsterName = "Brachiodemon"
+
+-- Registering encounters, stages and move events
+for raidNumber, raid in ipairs(SoulWarQuest.claustrophobicInfernoRaids) do
+ -- Registering encounter
+ local raidName = string.format("Claustrophobic Inferno Raid %d", raidNumber)
+ local encounter = Encounter(raidName, {
+ zone = raid.getZone(),
+ timeToSpawnMonsters = "3s",
+ })
+
+ local spawnTimes = SoulWarQuest.claustrophobicInfernoRaids.suriviveTime / SoulWarQuest.claustrophobicInfernoRaids.spawnTime
+
+ -- Registering encounter stages
+ for i = 1, spawnTimes do
+ encounter
+ :addSpawnMonsters({
+ {
+ name = spawnMonsterName,
+ positions = raid.spawns,
+ },
+ })
+ :autoAdvance(SoulWarQuest.claustrophobicInfernoRaids.spawnTime * 1000)
+ end
+
+ function encounter:onReset(position)
+ encounter:removeMonsters()
+ addEvent(function(zone)
+ zone:refresh()
+ zone:removePlayers()
+ end, SoulWarQuest.claustrophobicInfernoRaids.timeToKick * 1000, raid.getZone())
+ logger.debug("{} has ended", raidName)
+ end
+
+ encounter:register()
+
+ -- Registering move event
+ local raidMoveEvent = MoveEvent()
+
+ function raidMoveEvent.onStepIn(creature, item, position, fromPosition)
+ if not creature:getPlayer() then
+ return true
+ end
+ if fromPosition.y == position.y - (raidNumber % 2 ~= 0 and -1 or 1) then -- if player comes from the raid zone don't start the raid
+ return
+ end
+ logger.debug("{} has started", raidName)
+ encounter:start()
+ return true
+ end
+
+ for _, pos in pairs(raid.sandTimerPositions) do
+ raidMoveEvent:position(pos)
+ end
+
+ raidMoveEvent:register()
+end
diff --git a/data-otservbr-global/scripts/quests/soul_war/moveevent-soul_war_entrances.lua b/data-otservbr-global/scripts/quests/soul_war/moveevent-soul_war_entrances.lua
new file mode 100644
index 00000000000..0d2bd4fffec
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/moveevent-soul_war_entrances.lua
@@ -0,0 +1,147 @@
+local positionsTable = {
+ -- Hunts
+ [Position(33615, 31422, 10)] = Position(34009, 31014, 9), -- hunt infernal demon
+ [Position(33618, 31422, 10)] = Position(33972, 31041, 11), -- hunt rotten
+ [Position(33621, 31422, 10)] = Position(33894, 31019, 8), -- hunt bony sea devil
+ [Position(33624, 31422, 10)] = Position(33858, 31831, 3), -- hunt cloak
+ [Position(33627, 31422, 10)] = Position(33887, 31188, 10), -- hunt many faces
+}
+
+local soul_war_entrances = MoveEvent()
+
+function soul_war_entrances.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return
+ end
+
+ if player:getLevel() < 250 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need level 250 to enter here.")
+ player:teleportTo(fromPosition, true)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return
+ end
+
+ -- Check if player has access to teleport from Flickering Soul npc: "hi/task/yes"
+ local soulWarQuest = player:soulWarQuestKV()
+ if not soulWarQuest:get("teleport-access") then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Your soul does not yet resonate with the frequency required to enter here.")
+ player:teleportTo(fromPosition, true)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return
+ end
+
+ for position, destination in pairs(positionsTable) do
+ if position == player:getPosition() then
+ fromPosition:sendMagicEffect(CONST_ME_TELEPORT)
+ player:teleportTo(destination)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ break
+ end
+ end
+
+ return true
+end
+
+for key, value in pairs(positionsTable) do
+ soul_war_entrances:position(key)
+end
+
+soul_war_entrances:register()
+
+local soul_war_megalomania_entrance = MoveEvent()
+
+function soul_war_megalomania_entrance.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ if player:getLevel() < 250 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You are not allowed to enter here.")
+ player:teleportTo(fromPosition, true)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return false
+ end
+
+ local text = ""
+ local soulWarCount = 0
+ for bossName, completed in pairs(SoulWarQuest.miniBosses) do
+ if soulWarQuest:get(bossName) == completed then
+ soulWarCount = soulWarCount + 1
+ else
+ text = text .. "\n" .. bossName
+ end
+ end
+
+ if soulWarCount < 5 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You still need to defeat:" .. text)
+ player:teleportTo(fromPosition, true)
+ return false
+ end
+
+ return true
+end
+
+soul_war_megalomania_entrance:position({ x = 33611, y = 31430, z = 10 })
+soul_war_megalomania_entrance:register()
+
+local claustrophobicInfernoTeleportPositions = {
+ [Position(34022, 31091, 11)] = Position(33685, 31599, 14),
+}
+
+local claustrophobicInfernoTeleports = MoveEvent()
+
+function claustrophobicInfernoTeleports.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ for tablePosition, toPosition in pairs(claustrophobicInfernoTeleportPositions) do
+ if tablePosition == position then
+ player:teleportTo(toPosition)
+ toPosition:sendMagicEffect(CONST_ME_TELEPORT)
+ break
+ end
+ end
+
+ return true
+end
+
+for key, value in pairs(claustrophobicInfernoTeleportPositions) do
+ claustrophobicInfernoTeleports:position(key)
+end
+
+claustrophobicInfernoTeleports:register()
+
+local goshnarSpiteEntrance = MoveEvent()
+
+function goshnarSpiteEntrance.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ local killCount = soulWarQuest:get("hazardous-phantom-death") or 0
+ if killCount < 20 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have killed " .. killCount .. " and need to kill 20 Hazardous Phantoms")
+ player:teleportTo(fromPosition, true)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return false
+ end
+
+ if position == SoulWarQuest.goshnarSpiteEntrancePosition.fromPos then
+ player:teleportTo(SoulWarQuest.goshnarSpiteEntrancePosition.toPos)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+ end
+
+ return false
+end
+
+goshnarSpiteEntrance:position(SoulWarQuest.goshnarSpiteEntrancePosition.fromPos)
+goshnarSpiteEntrance:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/actions_portal_reward_soulwar.lua b/data-otservbr-global/scripts/quests/soul_war/moveevent-teleport_entrance_reward.lua
similarity index 78%
rename from data-otservbr-global/scripts/quests/soul_war/actions_portal_reward_soulwar.lua
rename to data-otservbr-global/scripts/quests/soul_war/moveevent-teleport_entrance_reward.lua
index cf999f89c7b..7aaff112056 100644
--- a/data-otservbr-global/scripts/quests/soul_war/actions_portal_reward_soulwar.lua
+++ b/data-otservbr-global/scripts/quests/soul_war/moveevent-teleport_entrance_reward.lua
@@ -1,10 +1,14 @@
local portalReward = MoveEvent()
+
function portalReward.onStepIn(creature, item, position, fromPosition)
local player = creature:getPlayer()
if not player then
return false
end
- if player:getStorageValue(Storage.Quest.U12_40.SoulWar.GoshnarMegalomaniaKilled) < 1 then
+
+ local soulWarQuest = player:soulWarQuestKV()
+ -- Checks if the boss has already been defeated
+ if not soulWarQuest:get("goshnar's-megalomania-killed") then
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Only warriors who defeated Goshnar's Megalomania can access this area.")
player:teleportTo(fromPosition, true)
return false
@@ -15,6 +19,5 @@ function portalReward.onStepIn(creature, item, position, fromPosition)
return true
end
-portalReward:type("stepin")
portalReward:position({ x = 33621, y = 31416, z = 10 })
portalReward:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/soul_war_mechanics.lua b/data-otservbr-global/scripts/quests/soul_war/soul_war_mechanics.lua
new file mode 100644
index 00000000000..6a60195277a
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/soul_war_mechanics.lua
@@ -0,0 +1,1081 @@
+local login = CreatureEvent("SoulWarLogin")
+
+function login.onLogin(player)
+ player:registerEvent("GoshnarsHatredBuff")
+ player:resetTaints()
+ player:resetGoshnarSymbolTormentCounter()
+ return true
+end
+
+login:register()
+
+-- Goshnar's Malice reflection (100%) of physical and death damage
+local goshnarsMaliceReflection = CreatureEvent("Goshnar's-Malice")
+
+function goshnarsMaliceReflection.onHealthChange(creature, attacker, primaryDamage, primaryType, secondaryDamage, secondaryType, origin)
+ if not attacker then
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+ end
+
+ local player = attacker:getPlayer()
+ if player then
+ if primaryDamage > 0 and (primaryType == COMBAT_PHYSICALDAMAGE or primaryType == COMBAT_DEATHDAMAGE) then
+ player:addHealth(-primaryDamage)
+ end
+ if secondaryDamage > 0 and (secondaryType == COMBAT_PHYSICALDAMAGE or secondaryType == COMBAT_DEATHDAMAGE) then
+ player:addHealth(-secondaryDamage)
+ end
+ end
+
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+end
+
+goshnarsMaliceReflection:register()
+
+local soulCageReflection = CreatureEvent("SoulCageHealthChange")
+
+function soulCageReflection.onHealthChange(creature, attacker, primaryDamage, primaryType, secondaryDamage, secondaryType, origin)
+ local player = attacker:getPlayer()
+ if player then
+ if primaryDamage > 0 then
+ player:addHealth(-primaryDamage * 0.1)
+ end
+ if secondaryDamage > 0 then
+ player:addHealth(-secondaryDamage * 0.1)
+ end
+ end
+
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+end
+
+soulCageReflection:register()
+
+local soulCageDeath = CreatureEvent("SoulCageDeath")
+
+function soulCageDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ if not creature or creature:isPlayer() or creature:getMaster() then
+ return true
+ end
+
+ addEvent(SpawnSoulCage, 23000)
+end
+
+soulCageDeath:register()
+
+local fourthTaintBossesDeath = CreatureEvent("FourthTaintBossesPrepareDeath")
+
+function fourthTaintBossesDeath.onPrepareDeath(creature, killer, realDamage)
+ if not creature or not killer:getPlayer() then
+ return true
+ end
+
+ if creature:getHealth() - realDamage < 1 then
+ if killer:getTaintNameByNumber(4) then
+ local isInZone = killer:getSoulWarZoneMonster()
+ if isInZone ~= nil then
+ -- 10% of chance to heal
+ if math.random(1, 10) == 1 then
+ creature:say("Health restored by the mystic powers of Zarganash!")
+ creature:addHealth(creature:getMaxHealth())
+ end
+ end
+ end
+ end
+ return true
+end
+
+fourthTaintBossesDeath:register()
+
+local bossesDeath = CreatureEvent("SoulWarBossesDeath")
+
+function bossesDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ local bossName = creature:getName()
+ if SoulWarQuest.miniBosses[bossName] then
+ local killers = creature:getKillers(true)
+ for i, killerPlayer in ipairs(killers) do
+ logger.debug("Player {} killed the boss.", killerPlayer:getName())
+ local soulWarQuest = killerPlayer:soulWarQuestKV()
+ -- Checks if the boss has already been defeated
+ if not soulWarQuest:get(bossName) then
+ local firstTaintTime = soulWarQuest:get("firstTaintTime")
+ if not firstTaintTime then
+ local currentTime = os.time()
+ soulWarQuest:set("firstTaintTime", currentTime)
+ end
+
+ soulWarQuest:set(bossName, true) -- Mark the boss as defeated
+ -- Adds the next taint in the sequence that the player does not already have
+ killerPlayer:addNextTaint()
+ end
+ end
+ end
+end
+
+bossesDeath:register()
+
+fourthTaintBossesDeath:register()
+
+local lastUse = 0
+local cooldown = 30
+
+local mirrorImageCreation = Action()
+function mirrorImageCreation.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ local currentTime = os.time()
+ local timePassed = currentTime - lastUse
+ if timePassed >= cooldown or lastUse == 0 then
+ Game.createMonster("Mirror Image", player:getPosition())
+ lastUse = currentTime
+ item:transform(33783)
+ else
+ local timeLeft = cooldown - timePassed
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need to wait " .. timeLeft .. " second(s) to use this item again.")
+ end
+
+ return true
+end
+
+mirrorImageCreation:id(33782)
+mirrorImageCreation:register()
+
+local mirroredNightmareApparitionDeath = CreatureEvent("MirroredNightmareBossAccess")
+
+function mirroredNightmareApparitionDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ local creatureName = creature:getName()
+ if table.contains(SoulWarQuest.apparitionNames, creatureName) then
+ local damageMap = creature:getMonster():getDamageMap()
+ for key, _ in pairs(damageMap) do
+ local player = Player(key)
+ if player then
+ local soulWarQuest = player:soulWarQuestKV()
+ local currentCount = soulWarQuest:get(creatureName) or 0
+ soulWarQuest:set(creatureName, currentCount + 1)
+ end
+ end
+ end
+end
+
+mirroredNightmareApparitionDeath:register()
+
+-- Check mirrored nightmare boss access
+local goshnarGreedEntrance = MoveEvent()
+
+function goshnarGreedEntrance.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ local hasAccess = true
+ local message = "Progress towards Mirrored Nightmare boss access:\n"
+
+ for _, apparitionName in pairs(SoulWarQuest.apparitionNames) do
+ local count = soulWarQuest:get(apparitionName) or 0
+ if count < SoulWarQuest.requiredCountPerApparition then
+ hasAccess = false
+ message = message .. apparitionName .. ": " .. count .. "/" .. SoulWarQuest.requiredCountPerApparition .. " kills\n"
+ else
+ message = message .. apparitionName .. ": Access achieved!\n"
+ end
+ end
+
+ if not hasAccess then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, message)
+ player:teleportTo(fromPosition)
+ return false
+ end
+
+ player:teleportTo(SoulWarQuest.goshnarsGreedAccessPosition.to)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+end
+
+goshnarGreedEntrance:position(SoulWarQuest.goshnarsGreedAccessPosition.from)
+goshnarGreedEntrance:register()
+
+local greedMonsterDeath = CreatureEvent("GreedMonsterDeath")
+
+function greedMonsterDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ local createMonsterPosition = GreedMonsters[creature:getName()]
+ if creature:getName() == "Greedbeast" then
+ GreedbeastKills = GreedbeastKills + 1
+ end
+
+ CreateGoshnarsGreedMonster(creature:getName(), createMonsterPosition)
+end
+
+greedMonsterDeath:register()
+
+local checkTaint = TalkAction("!checktaint")
+
+function checkTaint.onSay(player, words, param)
+ local taintLevel = player:getTaintLevel()
+ local taintName = player:getTaintNameByNumber(taintLevel)
+ if taintLevel ~= nil and taintName ~= nil then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Your current taint level is: " .. taintLevel .. " name: " .. taintName)
+ else
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You currently have no taint.")
+ end
+
+ return true
+end
+
+checkTaint:groupType("normal")
+checkTaint:register()
+
+local setTaint = TalkAction("/settaint")
+
+function setTaint.onSay(player, words, param)
+ local split = param:split(",")
+ local target = Player(split[1])
+ if not target then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Player is offline")
+ return false
+ end
+
+ local taintLevel = split[2]:trim():lower()
+ local taintName = player:getTaintNameByNumber(tonumber(taintLevel), true)
+ if taintName ~= nil then
+ target:resetTaints(true)
+ target:soulWarQuestKV():set(taintName, true)
+ target:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You new taint level is: " .. taintLevel .. ", name: " .. taintName)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Added taint level: " .. taintLevel .. ", name: " .. taintName .. " to player: " .. target:getName())
+ target:setTaintIcon()
+ end
+end
+
+setTaint:separator(" ")
+setTaint:groupType("god")
+setTaint:register()
+
+local goshnarGreedTeleport = MoveEvent()
+
+function goshnarGreedTeleport.onStepIn(creature, item, position, fromPosition)
+ local creatureName = creature:getName()
+ if creatureName == "Greedbeast" then
+ return
+ end
+
+ local foundCreaturePosition = GreedMonsters[creatureName]
+ if not foundCreaturePosition then
+ return false
+ end
+
+ if item:getId() == 33791 then
+ creature:remove()
+ item:transform(33790)
+ position:sendMagicEffect(CONST_ME_MORTAREA)
+ CreateGoshnarsGreedMonster(creatureName, foundCreaturePosition)
+ end
+
+ return true
+end
+
+goshnarGreedTeleport:id(33790, 33791)
+goshnarGreedTeleport:register()
+
+local setTaint = TalkAction("/removetaint")
+
+function setTaint.onSay(player, words, param)
+ local split = param:split(",")
+ local target = Player(split[1])
+ if not target then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Player is offline")
+ return false
+ end
+
+ local taintLevel = split[2]:trim():lower()
+ local taintName = player:getTaintNameByNumber(tonumber(taintLevel))
+ if taintName ~= nil then
+ target:soulWarQuestKV():remove(taintName)
+ target:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You lose taint level: " .. taintLevel .. ", name: " .. taintName)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Removed taint level: " .. taintLevel .. ", name: " .. taintName .. " from player: " .. target:getName())
+ end
+end
+
+setTaint:separator(" ")
+setTaint:groupType("god")
+setTaint:register()
+
+local changeMap = TalkAction("/changeflowmap")
+
+function changeMap.onSay(player, words, param)
+ if param == "empty" then
+ Game.loadMap(SoulWarQuest.ebbAndFlow.mapsPath.empty)
+ elseif param == "inundate" then
+ Game.loadMap(SoulWarQuest.ebbAndFlow.mapsPath.inundate)
+ elseif param == "ebb" then
+ Game.loadMap(SoulWarQuest.ebbAndFlowmapsPath.ebbFlow)
+ end
+end
+
+changeMap:separator(" ")
+changeMap:groupType("god")
+changeMap:register()
+
+local hazardousPhantomDeath = CreatureEvent("HazardousPhantomDeath")
+
+function hazardousPhantomDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ local killers = creature:getKillers(true)
+ for i, killerPlayer in ipairs(killers) do
+ -- Checks if the killer is a player
+ if killerPlayer:isPlayer() then
+ local soulWarQuest = killerPlayer:soulWarQuestKV()
+ local deathCount = soulWarQuest:get("hazardous-phantom-death") or 0
+ -- Checks that the death count has not yet reached the limit
+ if deathCount < SoulWarQuest.hardozousPanthomDeathCount then
+ -- Increases death count
+ soulWarQuest:set("hazardous-phantom-death", deathCount + 1)
+ -- Send the count for the player
+ killerPlayer:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You killed " .. (deathCount + 1) .. " of " .. SoulWarQuest.hardozousPanthomDeathCount .. " Hazardous Panthom.")
+ end
+
+ if deathCount + 1 == SoulWarQuest.hardozousPanthomDeathCount then
+ killerPlayer:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You can now access the boss room.")
+ end
+ end
+ end
+end
+
+hazardousPhantomDeath:register()
+
+local weepingSoulCorpse = MoveEvent()
+
+local condition = Condition(CONDITION_OUTFIT)
+condition:setOutfit(SoulWarQuest.waterElementalOutfit)
+condition:setTicks(14000)
+
+function weepingSoulCorpse.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return
+ end
+
+ if player:hasCondition(CONDITION_OUTFIT) then
+ return
+ end
+
+ local monster = Creature("Goshnar's Spite")
+ if monster then
+ local chance = math.random(100)
+ if chance <= SoulWarQuest.goshnarsSpiteHealChance then
+ local healAmount = math.floor(monster:getMaxHealth() * (SoulWarQuest.goshnarsSpiteHealPercentage / 100))
+ -- Heal percentage of the maximum health
+ monster:addHealth(healAmount)
+ logger.debug("Goshnar's Spite was healed to 10% of its maximum health.")
+ end
+ end
+
+ item:remove()
+ player:addCondition(condition)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You are soaked by tears of the weeping soul!")
+ return true
+end
+
+weepingSoulCorpse:id(SoulWarQuest.weepingSoulCorpseId)
+weepingSoulCorpse:register()
+
+local function removeSearingFire(position)
+ local tile = Tile(position)
+ if tile then
+ local fire = tile:getItemById(SoulWarQuest.searingFireId)
+ if fire then
+ local monster = Creature("Goshnar's Spite")
+ if monster then
+ monster:addDefense(SoulWarQuest.goshnarsSpiteIncreaseDefense)
+ logger.debug("Found Goshnar's Spite on boss zone, adding defense.")
+ end
+ fire:remove()
+ end
+ end
+end
+
+local goshnarSpiteFire = GlobalEvent("CreateGoshnarSpiteFire")
+
+function goshnarSpiteFire.onThink(interval)
+ local randomIndex = math.random(#SoulWarQuest.goshnarsSpiteFirePositions) -- Choose a random index
+ local firePosition = SoulWarQuest.goshnarsSpiteFirePositions[randomIndex] -- Get the corresponding position
+ local tile = Tile(firePosition)
+ if tile then
+ local fire = Game.createItem(SoulWarQuest.searingFireId, 1, firePosition)
+ if fire then
+ addEvent(removeSearingFire, SoulWarQuest.timeToRemoveSearingFire * 1000, firePosition)
+ end
+ end
+
+ return true
+end
+
+goshnarSpiteFire:interval(SoulWarQuest.timeToCreateSearingFire * 1000)
+goshnarSpiteFire:register()
+
+local goshnarSpiteSoulFire = MoveEvent()
+
+function goshnarSpiteSoulFire.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return
+ end
+
+ local tile = Tile(position)
+ if not tile then
+ return
+ end
+
+ local searingFire = tile:getItemById(SoulWarQuest.searingFireId)
+ if not searingFire then
+ return
+ end
+
+ local soulWarQuest = player:soulWarQuestKV()
+ local lastSteppedTime = soulWarQuest:get("goshnar-spite-fire") or 0
+ local currentTime = os.time()
+
+ if lastSteppedTime + SoulWarQuest.cooldownToStepOnSearingFire > currentTime then
+ local remainingTime = lastSteppedTime + SoulWarQuest.cooldownToStepOnSearingFire - currentTime
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "His soul won't need to recover again! You need wait " .. remainingTime .. " seconds.")
+ return true
+ end
+
+ addEvent(function(playerId)
+ local eventPlayer = Player(playerId)
+ if eventPlayer then
+ eventPlayer:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Your soul has recovered!")
+ end
+ end, SoulWarQuest.cooldownToStepOnSearingFire * 1000, player:getId())
+
+ soulWarQuest:set("goshnar-spite-fire", currentTime)
+ searingFire:remove()
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The soul fire was stomped out in time! Your soul will now have to recover before you can do this again.")
+
+ return true
+end
+
+for _, pos in pairs(SoulWarQuest.goshnarsSpiteFirePositions) do
+ goshnarSpiteSoulFire:position(pos)
+end
+
+goshnarSpiteSoulFire:register()
+
+local ebbAndFlowBoatTeleports = MoveEvent()
+
+function ebbAndFlowBoatTeleports.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player or not SoulWarQuest.ebbAndFlow.isActive() then
+ return
+ end
+
+ for _, pos in pairs(SoulWarQuest.ebbAndFlowBoatTeleportPositions) do
+ if Position(pos.register) == position then
+ player:teleportTo(pos.teleportTo)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+ end
+ end
+end
+
+for _, pos in pairs(SoulWarQuest.ebbAndFlowBoatTeleportPositions) do
+ ebbAndFlowBoatTeleports:position(pos.register)
+end
+ebbAndFlowBoatTeleports:register()
+
+local ebbAndFlowDoor = Action()
+
+function ebbAndFlowDoor.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ if SoulWarQuest.ebbAndFlow.isActive() then
+ return false
+ end
+
+ -- Determines whether the player is north or south of the door
+ local playerPosition = player:getPosition()
+ local destination = Position(toPosition.x, toPosition.y, toPosition.z)
+ if playerPosition.y < toPosition.y then
+ -- Player is north, move south
+ destination.y = toPosition.y + 1
+ else
+ -- Player is south (or at the same y position), moves north
+ destination.y = toPosition.y - 1
+ end
+
+ player:teleportTo(destination)
+ destination:sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+end
+
+ebbAndFlowDoor:id(SoulWarQuest.ebbAndFlow.doorId)
+ebbAndFlowDoor:register()
+
+local rottenWastelandShrines = Action()
+
+function rottenWastelandShrines.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ local soulWarQuest = player:soulWarQuestKV()
+ local shrineUsed = soulWarQuest:get("rotten-wasterland-activated-shrine-id") or 0
+ if shrineUsed == item:getId() then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have already activated this shrine.")
+ return true
+ end
+
+ local activatedShrinesCount = soulWarQuest:get("rotten-wasterland-activated-shrine-count") or 0
+ if activatedShrinesCount >= 4 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have already activated all the shrines.")
+ return true
+ end
+
+ soulWarQuest:set("rotten-wasterland-activated-shrine-id", item:getId())
+
+ soulWarQuest:set("rotten-wasterland-activated-shrine-count", activatedShrinesCount + 1)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have activated this shrine.")
+ return true
+end
+
+for itemId, position in pairs(SoulWarQuest.rottenWastelandShrines) do
+ rottenWastelandShrines:id(itemId)
+end
+
+rottenWastelandShrines:register()
+
+local goshnarsHatredAccess = Action()
+
+function goshnarsHatredAccess.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ local soulWarQuest = player:soulWarQuestKV()
+ local activatedShrineCount = soulWarQuest:get("rotten-wasterland-activated-shrine-count") or 0
+ if activatedShrineCount < 4 then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You still need to activate all the shrines.")
+ return true
+ end
+
+ player:teleportTo(SoulWarQuest.goshnarsHatredAccessPosition.to)
+ player:getPosition():sendMagicEffect(CONST_ME_TELEPORT)
+ return true
+end
+
+goshnarsHatredAccess:position(SoulWarQuest.goshnarsHatredAccessPosition.from)
+goshnarsHatredAccess:register()
+
+local goshnarsHatredSorrow = Action()
+
+function goshnarsHatredSorrow.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ if not target then
+ return
+ end
+
+ if not table.contains(SoulWarQuest.burningHatredMonsters, target:getName()) then
+ logger.error("Player {} tried to use the item on a non-burning hatred monster.", player:getName())
+ return
+ end
+
+ item:remove()
+ local actualTime = SoulWarQuest.kvBurning:get("time") or 0
+ SoulWarQuest.kvBurning:set("time", actualTime + 10)
+ logger.debug("Player {} used the item on the monster {}, oldTime {}, newTime {}.", player:getName(), target:getName(), actualTime, actualTime + 10)
+ player:say("The flame of hatred is doused!", TALKTYPE_MONSTER_SAY, 0, 0, target:getPosition())
+ return true
+end
+
+goshnarsHatredSorrow:id(SoulWarQuest.goshnarsHatredSorrowId)
+goshnarsHatredSorrow:register()
+
+local burningChangeForm = CreatureEvent("BurningChangeForm")
+
+function burningChangeForm.onThink(creature)
+ if not creature or not creature:getMonster() then
+ return true
+ end
+
+ local monster = creature:getMonster()
+ local currentTime = SoulWarQuest.kvBurning:get("time") or 0
+ if currentTime == 0 then
+ SoulWarQuest.kvBurning:set("time", 180)
+ return true
+ end
+
+ SoulWarQuest.kvBurning:set("time", currentTime - 1)
+
+ logger.debug("Burning transformation decreased to time : {}", currentTime)
+ for _, transformation in ipairs(SoulWarQuest.burningTransformations) do
+ local timeTransformation, newType = unpack(transformation)
+ if currentTime == timeTransformation and monster:getName() ~= newType then
+ monster:setType(newType, true)
+ logger.debug("Changing monster to {} on currentTime {}.", newType, currentTime)
+
+ if newType == "Ashes of Burning Hatred" then
+ monster:say("The fire of hatred fuels and empowers Goshnar's Hate!", TALKTYPE_MONSTER_SAY, 0, 0, monster:getPosition())
+ local boss = Creature("Goshnar's Hatred")
+ if boss then
+ logger.debug("Increasing hatred damage multiplier.")
+ boss:increaseHatredDamageMultiplier(10)
+ end
+ logger.debug("Beginning of the burning transformation cycle.")
+ end
+ break
+ end
+ end
+
+ return true
+end
+
+burningChangeForm:register()
+
+local goshnarsHatredBuff = CreatureEvent("GoshnarsHatredBuff")
+
+function goshnarsHatredBuff.onHealthChange(creature, attacker, primaryDamage, primaryType, secondaryDamage, secondaryType, origin)
+ -- Ensure both attacker and creature are valid and the creature is "Goshnar's Hatred"
+ if creature then
+ -- Check if the attacker is a player and the creature is being hit
+ if attacker and creature:isMonster() and attacker:isPlayer() and (creature:getName() == "Goshnar's Hatred" or creature:getName() == "Goshnar's Megalomania") then
+ local defenseMultiplier = creature:getHatredDamageMultiplier()
+ if defenseMultiplier > 0 then
+ -- Apply the defense multiplier
+ creature:addDefense(defenseMultiplier)
+ logger.debug("Adding defense to {}.", creature:getName())
+ end
+ -- Check if the attacker is a monster and the player is being hit
+ elseif attacker and creature:isPlayer() and attacker:isMonster() and (attacker:getName() == "Goshnar's Hatred" or creature:getName() == "Goshnar's Megalomania") then
+ local damageMultiplier = attacker:getHatredDamageMultiplier()
+ if damageMultiplier > 0 then
+ local multip = 1 + (damageMultiplier / 100)
+ logger.debug("Adding damage: {} to {}.", multip, attacker:getName())
+ -- Return modified damage values
+ return primaryDamage * multip, primaryType, secondaryDamage, secondaryType
+ end
+ end
+ end
+
+ -- Return original damage values if no conditions are met
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+end
+
+goshnarsHatredBuff:register()
+
+local condensedRemorse = MoveEvent()
+
+function condensedRemorse.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local soulWarKV = player:soulWarQuestKV()
+ local remorseCount = soulWarKV:get("condensed-remorse") or 0
+ soulWarKV:set("condensed-remorse", remorseCount + 1)
+ if remorseCount + 1 == 2 then
+ player:resetGoshnarSymbolTormentCounter()
+ player:say("The remorse calms your dread!", TALKTYPE_MONSTER_SAY, 0, 0, item:getPosition())
+ player:getPosition():sendMagicEffect(CONST_ME_HOLYAREA)
+ soulWarKV:remove("condensed-remorse")
+ end
+
+ item:remove()
+ return true
+end
+
+condensedRemorse:id(SoulWarQuest.condensedRemorseId)
+condensedRemorse:register()
+
+local furiousCraterAccess = EventCallback("FuriousCraterAccessDropLoot")
+
+function furiousCraterAccess.monsterOnDropLoot(monster, corpse)
+ if not monster or not corpse then
+ return
+ end
+
+ local player = Player(corpse:getCorpseOwner())
+ if not player or not player:canReceiveLoot() then
+ return
+ end
+
+ local mType = monster:getType()
+ if not mType then
+ return
+ end
+
+ if not table.contains(SoulWarQuest.pulsatingEnergyMonsters, mType:getName()) then
+ return
+ end
+
+ Game.createItem(SoulWarQuest.pulsatingEnergyId, 1, monster:getPosition())
+end
+
+furiousCraterAccess:register()
+
+local pulsatingEnergy = MoveEvent()
+
+function pulsatingEnergy.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ local kv = player:pulsatingEnergyKV()
+ local energyCount = kv:get("access-counter") or 0
+ energyCount = energyCount + 1
+ kv:set("access-counter", energyCount)
+
+ logger.debug("Player {} stepped on a pulsating energy, current count: {}", player:getName(), energyCount)
+
+ local firstFloorAccess = kv:get("first-floor-access") or false
+ local secondFloorAccess = kv:get("second-floor-access") or false
+ local thirdFloorAccess = kv:get("third-floor-access") or false
+ if thirdFloorAccess then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You've already gained access to fight with the Goshnar's Cruelty.")
+ return true
+ end
+
+ if energyCount >= 40 and not firstFloorAccess then
+ kv:set("access-counter", 0)
+ kv:set("first-floor-access", true)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You've gained access to the first floor. Continue collecting Pulsating Energies to gain further access.")
+ end
+
+ if energyCount >= 55 and not secondFloorAccess then
+ kv:set("access-counter", 0)
+ kv:set("second-floor-access", true)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You've gained access to the second floor. Continue collecting Pulsating Energies to gain further access.")
+ end
+
+ if energyCount >= 70 and not thirdFloorAccess then
+ kv:set("access-counter", 0)
+ kv:set("third-floor-access", true)
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You've gained access to the third floor. You can now fight with the Goshnar's Cruelty.")
+ end
+
+ item:remove()
+ return true
+end
+
+pulsatingEnergy:id(SoulWarQuest.pulsatingEnergyId)
+pulsatingEnergy:register()
+
+local pulsatingEnergyTeleportAccess = MoveEvent()
+
+function pulsatingEnergyTeleportAccess.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return false
+ end
+
+ for _, posData in pairs(SoulWarQuest.goshnarsCrueltyTeleportRoomPositions) do
+ if posData.from == position then
+ local kv = player:pulsatingEnergyKV()
+ local hasAccess = kv:get(posData.access) or false
+ local energyCount = kv:get("access-counter") or 0
+ local energiesNeeded = posData.count - energyCount
+ if not hasAccess then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You don't have access to this floor yet. You have collected " .. energyCount .. "/" .. posData.count .. ", and need " .. energiesNeeded .. " more pulsating energies to gain access.")
+ player:teleportTo(fromPosition, true)
+ fromPosition:sendMagicEffect(CONST_ME_TELEPORT)
+ else
+ player:teleportTo(posData.to)
+ posData.to:sendMagicEffect(CONST_ME_TELEPORT)
+ end
+
+ break
+ end
+ end
+
+ return true
+end
+
+for _, positions in pairs(SoulWarQuest.goshnarsCrueltyTeleportRoomPositions) do
+ pulsatingEnergyTeleportAccess:position(positions.from)
+end
+
+pulsatingEnergyTeleportAccess:register()
+
+local cloakOfTerrorHealthLoss = CreatureEvent("CloakOfTerrorHealthLoss")
+
+function cloakOfTerrorHealthLoss.onHealthChange(creature, attacker, primaryDamage, primaryType, secondaryDamage, secondaryType, origin)
+ if not creature or not attacker then
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+ end
+
+ if attacker:getPlayer() and primaryDamage > 0 or secondaryDamage > 0 then
+ local position = creature:getPosition()
+ local tile = Tile(position)
+ if tile then
+ if not tile:getItemById(SoulWarQuest.theBloodOfCloakTerrorIds[1]) then
+ Game.createItem(SoulWarQuest.theBloodOfCloakTerrorIds[1], 1, position)
+ end
+ end
+ end
+
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+end
+
+cloakOfTerrorHealthLoss:register()
+
+local theBloodOfCloakStep = MoveEvent()
+
+function theBloodOfCloakStep.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ -- If a player steps in blood, it takes damage
+ if player then
+ local damagePercentage = SoulWarQuest.poolDamagePercentages[item:getId()] or 0
+ local maxHealth = player:getMaxHealth()
+ local damage = maxHealth * damagePercentage
+
+ player:addHealth(-damage, COMBAT_ENERGYDAMAGE)
+ end
+
+ -- If a "Cloak of Terror" monster steps in blood, it heals itself
+ local monster = creature:getMonster()
+ if monster and monster:getName() == "Cloak of Terror" then
+ local healAmount = math.random(1500, 2000)
+ monster:addHealth(healAmount)
+ end
+
+ item:remove()
+
+ return true
+end
+
+for _, itemId in pairs(SoulWarQuest.theBloodOfCloakTerrorIds) do
+ theBloodOfCloakStep:id(itemId)
+end
+
+theBloodOfCloakStep:register()
+
+local greedyMaw = Action()
+
+function greedyMaw.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ if not item or not target then
+ logger.error("Greedy Maw action failed, item or target is nil.")
+ return false
+ end
+
+ if target:getId() == SoulWarQuest.greedyMawId then
+ local kv = player:soulWarQuestKV():scoped("furious-crater")
+ local cooldown = kv:get("greedy-maw-action") or 0
+ local currentTime = os.time()
+ if cooldown + SoulWarQuest.useGreedMawCooldown > currentTime then
+ local timeLeft = cooldown + SoulWarQuest.useGreedMawCooldown - currentTime
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need to wait " .. timeLeft .. " more seconds before using the greedy maw again.")
+ return true
+ end
+
+ kv:set("greedy-maw-action", currentTime)
+ local timeToIncreaseDefense = SoulWarQuest.timeToIncreaseCrueltyDefense
+ SoulWarQuest.kvSoulWar:set("greedy-maw-action", currentTime + timeToIncreaseDefense)
+ target:getPosition():sendMagicEffect(CONST_ME_DRAWBLOOD)
+ item:remove()
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Use the item again within " .. timeToIncreaseDefense .. " seconds, or the monster's defense will increase by 2 every " .. timeToIncreaseDefense .. " seconds.")
+ local goshnarsCruelty = Creature("Goshnar's Cruelty")
+ if goshnarsCruelty then
+ local mtype = goshnarsCruelty:getType()
+ if not mtype then
+ logger.error("Greedy Maw action failed, Goshnar's Cruelty has no type.")
+ return false
+ end
+
+ -- If the defense of Goshnar's Cruelty is higher than the default defense, decrease it by 2
+ if goshnarsCruelty:getDefense() > mtype:defense() then
+ logger.debug("Greedy Maw used on Goshnar's Cruelty, old defense {}", goshnarsCruelty:getDefense())
+ goshnarsCruelty:addDefense(-SoulWarQuest.goshnarsCrueltyDefenseChange)
+ logger.debug("Greedy Maw used on Goshnar's Cruelty, new defense {}", goshnarsCruelty:getDefense())
+ end
+
+ local defenseDrainValue = SoulWarQuest.kvSoulWar:get("goshnars-cruelty-defense-drain") or 0
+ if defenseDrainValue > 0 then
+ SoulWarQuest.kvSoulWar:set("goshnars-cruelty-defense-drain", defenseDrainValue - 1)
+ end
+ end
+ return true
+ end
+
+ return false
+end
+
+greedyMaw:id(SoulWarQuest.someMortalEssenceId)
+greedyMaw:register()
+
+local soulWarAspectOfPowerDeath = CreatureEvent("SoulWarAspectOfPowerDeath")
+
+function soulWarAspectOfPowerDeath.onDeath(creature)
+ local targetMonster = creature:getMonster()
+ if not targetMonster or targetMonster:getMaster() then
+ return
+ end
+
+ logger.debug("Aspect of Power died, checking if all are dead.")
+ local boss = Creature("Goshnar's Megalomania")
+ if boss and boss:getTypeName() == "Goshnar's Megalomania Purple" then
+ boss:increaseAspectOfPowerDeathCount()
+ end
+
+ local position = boss and boss:getPosition() or creature:getPosition()
+ addEvent(function(position)
+ local aspectMonster = Game.createMonster("Aspect of Power", position)
+ if aspectMonster then
+ local outfit = aspectMonster:getOutfit()
+ outfit.lookType = math.random(1303, 1307)
+ aspectMonster:setOutfit(outfit)
+ end
+ end, 5000, position)
+
+ return true
+end
+
+soulWarAspectOfPowerDeath:register()
+
+local madnessReduce = MoveEvent()
+
+function madnessReduce.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ item:getPosition():sendMagicEffect(CONST_ME_HOLYAREA)
+ item:remove()
+ if player and player:getGoshnarSymbolTormentCounter() > 0 then
+ player:resetGoshnarSymbolTormentCounter()
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "The ooze calms your dread but leaves you vulnerable to phantasmal attacks!")
+ return true
+ end
+
+ local creatureName = creature:getName()
+ if creatureName == "Lesser Splinter of Madness" or creatureName == "Greater Splinter of Madness" or creatureName == "Mighty Splinter of Madness" then
+ creature:remove()
+ item:transform(SoulWarQuest.cleansedSanityItemId)
+ end
+
+ return true
+end
+
+madnessReduce:id(SoulWarQuest.deadAspectOfPowerCorpseId)
+madnessReduce:register()
+
+local cleansedSanity = Action()
+
+function cleansedSanity.onUse(player, item, fromPosition, target, toPosition, isHotkey)
+ if not item or not target then
+ logger.error("Cleansed action failed, item or target is nil.")
+ return false
+ end
+
+ local kv = player:soulWarQuestKV():scoped("furious-crater")
+ local cooldown = kv:get("cleansed-sanity-action") or 0
+ local currentTime = os.time()
+ if cooldown + SoulWarQuest.useGreedMawCooldown > currentTime then
+ local timeLeft = cooldown + SoulWarQuest.useGreedMawCooldown - currentTime
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You need to wait " .. timeLeft .. " more seconds before using the cleansed again.")
+ return true
+ end
+
+ kv:set("cleansed-sanity-action", currentTime)
+ if target:getId() == SoulWarQuest.greedyMawId then
+ local timeToIncreaseDefense = SoulWarQuest.timeToIncreaseCrueltyDefense
+ SoulWarQuest.kvSoulWar:set("cleansed-sanity-action", currentTime + timeToIncreaseDefense)
+ target:getPosition():sendMagicEffect(CONST_ME_DRAWBLOOD)
+ item:remove()
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Use the item again within " .. timeToIncreaseDefense .. " seconds, or the monster's defense will increase every " .. timeToIncreaseDefense .. " seconds.")
+ local boss = Creature("Goshnar's Megalomania")
+ if boss then
+ local mtype = boss:getType()
+ if not mtype then
+ logger.error("Cleansed action failed, Goshnar's Megalomania has no type.")
+ return false
+ end
+
+ -- If the defense of Goshnar's Megalomania is higher than the default defense, decrease it by 2
+ if boss:getDefense() > mtype:defense() then
+ logger.debug("Cleansed used on Goshnar's Megalomania, old defense {}", boss:getDefense())
+ boss:addDefense(-SoulWarQuest.goshnarsCrueltyDefenseChange)
+ logger.debug("Cleansed used on Goshnar's Megalomania, new defense {}", boss:getDefense())
+ end
+ end
+ return true
+ end
+
+ return false
+end
+
+cleansedSanity:id(SoulWarQuest.cleansedSanityItemId)
+cleansedSanity:register()
+
+local necromanticRemainsReduce = MoveEvent()
+
+function necromanticRemainsReduce.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return
+ end
+
+ player:removeGoshnarSymbolTormentCounter(5)
+ item:remove()
+ position:sendMagicEffect(CONST_ME_HOLYAREA)
+ return true
+end
+
+necromanticRemainsReduce:id(SoulWarQuest.necromanticRemainsItemId)
+necromanticRemainsReduce:register()
+
+local necromanticFocusDeath = CreatureEvent("NecromanticFocusDeath")
+
+function necromanticFocusDeath.onDeath(creature)
+ local targetMonster = creature:getMonster()
+ if not targetMonster or targetMonster:getMaster() then
+ return
+ end
+
+ local position = targetMonster:getPosition()
+ addEvent(function()
+ position:increaseNecromaticMegalomaniaStrength()
+ end, 5 * 60 * 1000)
+
+ return true
+end
+
+necromanticFocusDeath:register()
+
+local megalomaniaDeath = CreatureEvent("MegalomaniaDeath")
+
+function megalomaniaDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUnjustified, mostDamageUnjustified)
+ local killers = creature:getKillers(true)
+ for i, killerPlayer in ipairs(killers) do
+ local soulWarQuest = killerPlayer:soulWarQuestKV()
+ -- Checks if the boss has already been defeated
+ if not soulWarQuest:get("goshnar's-megalomania-killed") then
+ soulWarQuest:set("goshnar's-megalomania-killed", true)
+ killerPlayer:sendTextMessage(MESSAGE_EVENT_ADVANCE, "You have defeated Goshnar's Megalomania. Report the 'task' to Flickering Soul and earn your outfit.")
+ end
+ end
+ return true
+end
+
+megalomaniaDeath:register()
+
+local teleportStepRemoveIcon = MoveEvent()
+
+function teleportStepRemoveIcon.onStepIn(creature, item, position, fromPosition)
+ local player = creature:getPlayer()
+ if not player then
+ return
+ end
+
+ player:resetGoshnarSymbolTormentCounter()
+ return true
+end
+
+local teleportPositions = {
+ Position(33713, 31642, 14),
+ Position(33743, 31606, 14),
+}
+
+for _, pos in pairs(teleportPositions) do
+ teleportStepRemoveIcon:position(pos)
+end
+
+teleportStepRemoveIcon:register()
+
+local goshnarsCrueltyBuff = CreatureEvent("GoshnarsCrueltyBuff")
+
+function goshnarsCrueltyBuff.onHealthChange(creature, attacker, primaryDamage, primaryType, secondaryDamage, secondaryType, origin)
+ if creature and creature:isMonster() and attacker:isPlayer() and creature:getName() == "Goshnar's Cruelty" then
+ local newValue = SoulWarQuest.kvSoulWar:get("goshnars-cruelty-defense-drain") or SoulWarQuest.goshnarsCrueltyDefenseChange
+ if newValue ~= 0 then
+ local multiplier = math.max(0, 1 - (newValue / 100))
+ return primaryDamage * multiplier, primaryType, secondaryDamage * multiplier, secondaryType
+ end
+ end
+
+ return primaryDamage, primaryType, secondaryDamage, secondaryType
+end
+
+goshnarsCrueltyBuff:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/spell-eye_beam.lua b/data-otservbr-global/scripts/quests/soul_war/spell-eye_beam.lua
new file mode 100644
index 00000000000..7a73979930a
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/spell-eye_beam.lua
@@ -0,0 +1,38 @@
+local combat = Combat()
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_DROWNDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_SMALLCLOUDS)
+
+combat:setArea(createCombatArea({
+ { 1 },
+ { 1 },
+ { 3 },
+}))
+
+function onTargetTile(cid, pos)
+ local tile = Tile(pos)
+ local target = tile:getTopCreature()
+ if tile then
+ if target then
+ if target:isMonster() and target:getName() == "Poor Soul" then
+ target:addHealth(-1000)
+ end
+ end
+ end
+ return true
+end
+
+combat:setCallback(CALLBACK_PARAM_TARGETTILE, "onTargetTile")
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, var)
+end
+
+spell:name("greedy eye beam")
+spell:words("greedy eye beam")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needDirection(true)
+spell:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_cruelty.lua b/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_cruelty.lua
new file mode 100644
index 00000000000..2bc321f24e0
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_cruelty.lua
@@ -0,0 +1,61 @@
+local areaSpell = {
+ { 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0 },
+ { 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 },
+ { 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0 },
+ { 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 },
+ { 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
+}
+
+local area = createCombatArea(areaSpell)
+
+local combat = Combat()
+combat:setArea(area)
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_FIREDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_FIREAREA)
+
+function onTargetTile(cid, pos)
+ local tile = Tile(pos)
+ if tile then
+ local target = tile:getTopCreature()
+ if target and target:isPlayer() then
+ target:addHealth(math.random(2300, 3000))
+ end
+ end
+ return true
+end
+
+combat:setCallback(CALLBACK_PARAM_TARGETTILE, "onTargetTile")
+
+local function delayedCastSpell(cid, var, targetId)
+ local creature = Creature(cid)
+ if not creature then
+ return
+ end
+
+ local target = Player(targetId)
+ if target then
+ combat:execute(creature, positionToVariant(target:getPosition()))
+ end
+end
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var, isHotkey)
+ return creature:applyZoneEffect(var, combat, "boss.goshnar's-cruelty")
+end
+
+spell:name("cruelty transform elemental")
+spell:words("cruelty transform elemental")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needDirection(true)
+spell:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_megalomania.lua b/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_megalomania.lua
new file mode 100644
index 00000000000..2906f213d74
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/spell-fire_beam_megalomania.lua
@@ -0,0 +1,54 @@
+local areaSpell = {
+ { 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0 },
+ { 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 },
+ { 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0 },
+ { 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0 },
+ { 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0 },
+ { 0, 0, 0, 0, 1, 1, 1, 1, 1, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
+}
+
+local area = createCombatArea(areaSpell)
+
+local combat = Combat()
+combat:setArea(area)
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_FIREDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_FIREAREA)
+
+function onTargetTile(cid, pos)
+ local tile = Tile(pos)
+ if tile then
+ local target = tile:getTopCreature()
+ if target and target:isPlayer() then
+ target:addHealth(math.random(2300, 3000))
+ end
+ end
+ return true
+end
+
+combat:setCallback(CALLBACK_PARAM_TARGETTILE, "onTargetTile")
+
+local config = {
+ outfit = { lookType = 242, lookHead = 0, lookBody = 0, lookLegs = 0, lookFeet = 0, lookAddons = 0 },
+ time = 7000,
+}
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var, isHotkey)
+ return creature:applyZoneEffect(var, combat, "boss.goshnar's-megalomania-purple")
+end
+
+spell:name("megalomania transform elemental")
+spell:words("megalomania transform elemental")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needDirection(true)
+spell:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/spell-megalomania_blue.lua b/data-otservbr-global/scripts/quests/soul_war/spell-megalomania_blue.lua
new file mode 100644
index 00000000000..d8a9c8e9875
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/spell-megalomania_blue.lua
@@ -0,0 +1,58 @@
+local area = {
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+ { 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
+}
+
+local createArea = createCombatArea(area)
+
+local combat = Combat()
+combat:setArea(createArea)
+
+local zone = Zone.getByName("boss.goshnar's-megalomania-purple")
+local zonePositions = zone:getPositions()
+
+function onTargetTile(creature, pos)
+ for _, pos in ipairs(zonePositions) do
+ local tile = Tile(pos)
+ if tile and tile:getGround() and tile:getGround():getId() ~= 409 then
+ local creature = tile:getTopCreature()
+ if creature then
+ local player = creature:getPlayer()
+ if player then
+ player:addHealth(-6000, COMBAT_DEATHDAMAGE)
+ end
+ end
+ end
+ end
+
+ pos:sendMagicEffect(CONST_ME_BLACKSMOKE)
+ return true
+end
+
+combat:setCallback(CALLBACK_PARAM_TARGETTILE, "onTargetTile")
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, positionToVariant(creature:getPosition()))
+end
+
+spell:name("megalomania blue")
+spell:words("megalomania blue")
+spell:isAggressive(true)
+spell:blockWalls(false)
+spell:needLearn(true)
+spell:register()
diff --git a/data-otservbr-global/scripts/quests/soul_war/spell-soulsnatcher.lua b/data-otservbr-global/scripts/quests/soul_war/spell-soulsnatcher.lua
new file mode 100644
index 00000000000..9292f10530f
--- /dev/null
+++ b/data-otservbr-global/scripts/quests/soul_war/spell-soulsnatcher.lua
@@ -0,0 +1,58 @@
+local combat = Combat()
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_PHYSICALDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_HOLYDAMAGE)
+
+combat:setArea(createCombatArea(CrossBeamArea3X2))
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, var)
+end
+
+spell:name("soulsnatcher-lifedrain-beam")
+spell:words("soulsnatcher-lifedrain-beam")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needDirection(true)
+spell:register()
+
+local combat = Combat()
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_PHYSICALDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_HOLYAREA)
+combat:setParameter(COMBAT_PARAM_DISTANCEEFFECT, CONST_ANI_HOLY)
+
+spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, var)
+end
+
+spell:name("soulsnatcher-lifedrain-missile")
+spell:words("soulsnatcher-lifedrain-missile")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needTarget(true)
+spell:register()
+
+-- Mana drain ball
+local combat = Combat()
+combat:setParameter(COMBAT_PARAM_TYPE, COMBAT_PHYSICALDAMAGE)
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_ENERGYAREA)
+
+combat:setArea(createCombatArea(AREA_CIRCLE3X3))
+
+spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, var)
+end
+
+spell:name("soulsnatcher-manadrain-ball")
+spell:words("soulsnatcher-manadrain-ball")
+spell:isAggressive(true)
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:register()
diff --git a/data-otservbr-global/scripts/world_changes/iron_servant_transformation.lua b/data-otservbr-global/scripts/world_changes/iron_servant_transformation.lua
index 65e6af5a3b2..7da5f5d1b99 100644
--- a/data-otservbr-global/scripts/world_changes/iron_servant_transformation.lua
+++ b/data-otservbr-global/scripts/world_changes/iron_servant_transformation.lua
@@ -1,4 +1,4 @@
-local ironServantTransformation = EventCallback()
+local ironServantTransformation = EventCallback("IronServantTransformationOnSpawn")
ironServantTransformation.monsterOnSpawn = function(monster, position)
if monster:getName():lower() ~= "iron servant replica" then
diff --git a/data-otservbr-global/startup/others/functions.lua b/data-otservbr-global/startup/others/functions.lua
index 8624dc8b268..362e2a38b97 100644
--- a/data-otservbr-global/startup/others/functions.lua
+++ b/data-otservbr-global/startup/others/functions.lua
@@ -28,7 +28,7 @@ function loadLuaMapAction(tablename)
if not value.itemId == false and tile:getItemCountById(value.itemId) == 0 then
logger.error("[loadLuaMapAction] - Wrong item id {} found", value.itemId)
logger.warn("Action id: {}, position {}", index, tile:getPosition():toString())
- break
+ goto continue
end
if value.itemId ~= false and tile:getItemCountById(value.itemId) > 0 then
@@ -49,6 +49,7 @@ function loadLuaMapAction(tablename)
tile:getGround():setAttribute(ITEM_ATTRIBUTE_ACTIONID, index)
end
end
+ ::continue::
end
end
end
@@ -64,12 +65,12 @@ function loadLuaMapUnique(tablename)
if not value.itemId == false and tile:getItemCountById(value.itemId) == 0 then
logger.error("[loadLuaMapUnique] - Wrong item id {} found", value.itemId)
logger.warn("Unique id: {}, position {}", index, tile:getPosition():toString())
- break
+ goto continue
end
if tile:getItemCountById(value.itemId) < 1 or value.itemId == false then
logger.warn("[loadLuaMapUnique] - Wrong item id {} found", value.itemId)
logger.warn("Unique id: {}, position {}, item id: wrong", index, tile:getPosition():toString())
- break
+ goto continue
end
item = tile:getItemById(value.itemId)
-- If he found the item, add the unique id
@@ -77,6 +78,8 @@ function loadLuaMapUnique(tablename)
item:setAttribute(ITEM_ATTRIBUTE_UNIQUEID, index)
end
end
+
+ ::continue::
end
end
@@ -91,7 +94,7 @@ function loadLuaMapSign(tablename)
if tile:getItemCountById(value.itemId) == 0 then
logger.error("[loadLuaMapSign] - Wrong item id {} found", value.itemId)
logger.warn("Sign id: {}, position {}, item id: wrong", index, tile:getPosition():toString())
- break
+ goto continue
end
if tile:getItemCountById(value.itemId) == 1 then
item = tile:getItemById(value.itemId)
@@ -101,6 +104,7 @@ function loadLuaMapSign(tablename)
item:setAttribute(ITEM_ATTRIBUTE_TEXT, value.text)
end
end
+ ::continue::
end
end
@@ -137,17 +141,18 @@ function loadLuaMapBookDocument(tablename)
totals[2] = totals[2] + 1
else
logger.warn("[loadLuaMapBookDocument] - Item not found! Index: {}, itemId: {}", index, value.itemId)
- break
+ goto continue
end
else
logger.warn("[loadLuaMapBookDocument] - Container not found! Index: {}, containerId: {}", index, value.containerId)
- break
+ goto continue
end
else
logger.warn("[loadLuaMapBookDocument] - Tile not found! Index: {}, position: x: {} y: {} z: {}", index, value.position.x, value.position.y, value.position.z)
- break
+ goto continue
end
end
+ ::continue::
end
if totals[1] == totals[2] then
logger.debug("Loaded {} books and documents in the map", totals[2])
diff --git a/data-otservbr-global/world/otservbr-house.xml b/data-otservbr-global/world/otservbr-house.xml
index f7e5cd52370..7eff23b4606 100644
--- a/data-otservbr-global/world/otservbr-house.xml
+++ b/data-otservbr-global/world/otservbr-house.xml
@@ -284,7 +284,7 @@
-
+
diff --git a/data-otservbr-global/world/otservbr-monster.xml b/data-otservbr-global/world/otservbr-monster.xml
index ef003deee1a..80c5f5d4ff3 100644
--- a/data-otservbr-global/world/otservbr-monster.xml
+++ b/data-otservbr-global/world/otservbr-monster.xml
@@ -34097,6 +34097,9 @@
+
+
+
@@ -34129,6 +34132,9 @@
+
+
+
@@ -34214,6 +34220,9 @@
+
+
+
@@ -161998,9 +162007,15 @@
+
+
+
+
+
+
@@ -162027,6 +162042,15 @@
+
+
+
+
+
+
+
+
+
@@ -162036,6 +162060,12 @@
+
+
+
+
+
+
@@ -162050,6 +162080,9 @@
+
+
+
@@ -162062,6 +162095,9 @@
+
+
+
@@ -162083,6 +162119,12 @@
+
+
+
+
+
+
@@ -162156,6 +162198,9 @@
+
+
+
@@ -162168,12 +162213,22 @@
+
+
+
+
+
+
+
+
+
+
@@ -162185,6 +162240,15 @@
+
+
+
+
+
+
+
+
+
@@ -162194,6 +162258,9 @@
+
+
+
@@ -162201,6 +162268,9 @@
+
+
+
diff --git a/data-otservbr-global/world/otservbr-npc.xml b/data-otservbr-global/world/otservbr-npc.xml
index 2011d333ad4..2b3f88a00d7 100644
--- a/data-otservbr-global/world/otservbr-npc.xml
+++ b/data-otservbr-global/world/otservbr-npc.xml
@@ -2460,6 +2460,9 @@
+
+
+
diff --git a/data-otservbr-global/world/otservbr-zones.xml b/data-otservbr-global/world/otservbr-zones.xml
index 4740d50385c..44cff897a42 100644
--- a/data-otservbr-global/world/otservbr-zones.xml
+++ b/data-otservbr-global/world/otservbr-zones.xml
@@ -1,4 +1,4 @@
-
+
diff --git a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-empty.otbm b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-empty.otbm
new file mode 100644
index 00000000000..9d3fa897425
Binary files /dev/null and b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-empty.otbm differ
diff --git a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/inundate.otbm b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-inundate.otbm
similarity index 80%
rename from data-otservbr-global/world/quest/soul_war/ebb_and_flow/inundate.otbm
rename to data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-inundate.otbm
index 1790545b34b..01f462c2f7a 100644
Binary files a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/inundate.otbm and b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow-inundate.otbm differ
diff --git a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow.otbm b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow.otbm
new file mode 100644
index 00000000000..5a8d5d16fef
Binary files /dev/null and b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/ebb-flow.otbm differ
diff --git a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/empty.otbm b/data-otservbr-global/world/quest/soul_war/ebb_and_flow/empty.otbm
deleted file mode 100644
index 70036d063b1..00000000000
Binary files a/data-otservbr-global/world/quest/soul_war/ebb_and_flow/empty.otbm and /dev/null differ
diff --git a/data/items/items.xml b/data/items/items.xml
index 4978cf0bdc4..8cec8f05ec6 100644
--- a/data/items/items.xml
+++ b/data/items/items.xml
@@ -62032,9 +62032,7 @@ hands of its owner. Granted by TibiaRoyal.com"/>
-
- -
-
-
+
-
@@ -62476,6 +62474,7 @@ hands of its owner. Granted by TibiaRoyal.com"/>
+
-
@@ -62891,28 +62890,25 @@ hands of its owner. Granted by TibiaRoyal.com"/>
-
+ -
+
+
+
-
- -
-
-
-
-
- -
-
-
- -
-
-
+
-
+
+
+
+
-
-
+
-
@@ -63066,6 +63062,10 @@ hands of its owner. Granted by TibiaRoyal.com"/>
+ -
+
+
+
-
@@ -63161,6 +63161,11 @@ hands of its owner. Granted by TibiaRoyal.com"/>
+ -
+
+
+
+
-
@@ -63180,6 +63185,7 @@ hands of its owner. Granted by TibiaRoyal.com"/>
+
-
@@ -63379,6 +63385,10 @@ hands of its owner. Granted by TibiaRoyal.com"/>
+ -
+
+
+
-
@@ -63525,6 +63535,19 @@ hands of its owner. Granted by TibiaRoyal.com"/>
-
+ -
+
+
+
+
+ -
+
+
+
+ -
+
+
+
-
@@ -64262,7 +64285,7 @@ hands of its owner. Granted by TibiaRoyal.com"/>
-
-
+
-
@@ -74750,12 +74773,31 @@ Granted by TibiaGoals.com"/>
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
-
+
-
+
-
@@ -74790,6 +74832,22 @@ Granted by TibiaGoals.com"/>
-
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
-
@@ -74808,6 +74866,15 @@ Granted by TibiaGoals.com"/>
+
+ -
+
+
+
+ -
+
+
+
-
@@ -76112,5 +76179,137 @@ Granted by TibiaGoals.com"/>
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+ -
+
+
+
+
diff --git a/data/libs/functions/boss_lever.lua b/data/libs/functions/boss_lever.lua
index b95bf7211b0..9cd577ee911 100644
--- a/data/libs/functions/boss_lever.lua
+++ b/data/libs/functions/boss_lever.lua
@@ -144,6 +144,7 @@ end
---@param player Player
---@return boolean
function BossLever:onUse(player)
+ local monsterName = MonsterType(self.name):getName()
local isParticipant = false
for _, v in ipairs(self.playerPositions) do
if Position(v.pos) == player:getPosition() then
@@ -161,7 +162,7 @@ function BossLever:onUse(player)
local zone = self:getZone()
if zone:countPlayers(IgnoredByMonsters) > 0 then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "There's already someone fighting with " .. self.name .. ".")
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "There's already someone fighting with " .. monsterName .. ".")
return true
end
@@ -173,14 +174,15 @@ function BossLever:onUse(player)
return true
end
- if creature:getLevel() < self.requiredLevel then
+ local isAccountNormal = creature:getAccountType() == ACCOUNT_TYPE_NORMAL
+ if isAccountNormal and creature:getLevel() < self.requiredLevel then
local message = "All players need to be level " .. self.requiredLevel .. " or higher."
creature:sendTextMessage(MESSAGE_EVENT_ADVANCE, message)
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, message)
return false
end
- if creature:getGroup():getId() < GROUP_TYPE_GOD and self:lastEncounterTime(creature) > os.time() then
+ if creature:getGroup():getId() < GROUP_TYPE_GOD and isAccountNormal and self:lastEncounterTime(creature) > os.time() then
local infoPositions = lever:getInfoPositions()
for _, posInfo in pairs(infoPositions) do
local currentPlayer = posInfo.creature
diff --git a/data/libs/functions/container.lua b/data/libs/functions/container.lua
index f932b37dfe4..824a12f5e02 100644
--- a/data/libs/functions/container.lua
+++ b/data/libs/functions/container.lua
@@ -20,7 +20,7 @@ function Container:addLoot(loot)
local countToAdd = math.min(remainingCount, stackSize)
local tmpItem = self:addItem(itemId, countToAdd, INDEX_WHEREEVER, FLAG_NOLIMIT)
if not tmpItem then
- logger.warn("Container:addLoot: failed to add stackable item: {}, to corpse {} with id {}", ItemType(itemId):getName(), self:getName(), self:getId())
+ logger.warn("Container:addLoot: failed to add stackable item: {} with id {}, to corpse {} with id {}", ItemType(itemId):getName(), itemId, self:getName(), self:getId())
goto continue
end
remainingCount = remainingCount - countToAdd
@@ -28,13 +28,13 @@ function Container:addLoot(loot)
elseif iType:getCharges() ~= 0 then
local tmpItem = self:addItem(itemId, item.count, INDEX_WHEREEVER, FLAG_NOLIMIT)
if not tmpItem then
- logger.warn("Container:addLoot: failed to add charge item: {}, to corpse {} with id {}", ItemType(itemId):getName(), self:getName(), self:getId())
+ logger.warn("Container:addLoot: failed to add charge item: {} with id {}, to corpse {} with id {}", ItemType(itemId):getName(), itemId, self:getName(), self:getId())
end
else
for i = 1, item.count do
local tmpItem = self:addItem(itemId, 1, INDEX_WHEREEVER, FLAG_NOLIMIT)
if not tmpItem then
- logger.warn("Container:addLoot: failed to add item: {}, to corpse {} with id {}", ItemType(itemId):getName(), self:getName(), self:getId())
+ logger.warn("Container:addLoot: failed to add item: {} with id {}, to corpse {} with id {}", ItemType(itemId):getName(), itemId, self:getName(), self:getId())
goto continue
end
diff --git a/data/libs/functions/creature.lua b/data/libs/functions/creature.lua
index be95a45144c..376f828c5b0 100644
--- a/data/libs/functions/creature.lua
+++ b/data/libs/functions/creature.lua
@@ -208,7 +208,7 @@ end
function Creature.getKillers(self, onlyPlayers)
local killers = {}
local inFightTicks = configManager.getNumber(configKeys.PZ_LOCKED)
- local timeNow = os.mtime()
+ local timeNow = systemTime()
local getCreature = onlyPlayers and Player or Creature
for cid, cb in pairs(self:getDamageMap()) do
local creature = getCreature(cid)
diff --git a/data/libs/functions/monster.lua b/data/libs/functions/monster.lua
index f26f2a5b9b4..0327daab3b3 100644
--- a/data/libs/functions/monster.lua
+++ b/data/libs/functions/monster.lua
@@ -204,7 +204,7 @@ do
return table.contains(equipmentTypes, t)
end
- function MonsterType.getBossReward(self, lootFactor, topScore, equipmentOnly, lootTable)
+ function MonsterType.getBossReward(self, lootFactor, topScore, equipmentOnly, lootTable, player)
if configManager.getNumber(configKeys.RATE_LOOT) <= 0 then
return lootTable or {}
end
@@ -221,6 +221,6 @@ do
end
return true
end,
- }, lootTable)
+ }, lootTable, player)
end
end
diff --git a/data/libs/functions/monstertype.lua b/data/libs/functions/monstertype.lua
index 168cab13109..a9f6fff59f7 100644
--- a/data/libs/functions/monstertype.lua
+++ b/data/libs/functions/monstertype.lua
@@ -1,7 +1,7 @@
-- return a dictionary of itemId => { count, gut }
---@param config { factor: number, gut: boolean, filter?: fun(itemType: ItemType, unique: boolean): boolean }
---@return LootItems
-function MonsterType:generateLootRoll(config, resultTable)
+function MonsterType:generateLootRoll(config, resultTable, player)
if configManager.getNumber(configKeys.RATE_LOOT) <= 0 then
return resultTable or {}
end
@@ -28,6 +28,11 @@ function MonsterType:generateLootRoll(config, resultTable)
end
local chance = item.chance
+ if iType:getId() == SoulWarQuest.bagYouDesireItemId then
+ result[item.itemId].chance = self:calculateBagYouDesireChance(player, chance)
+ logger.debug("Final chance for bag you desire: {}, original chance: {}", result[item.itemId].chance, chance)
+ end
+
if config.gut and iType:getType() == ITEM_TYPE_CREATUREPRODUCT then
chance = math.ceil((chance * GLOBAL_CHARM_GUT) / 100)
end
diff --git a/data/libs/functions/revscriptsys.lua b/data/libs/functions/revscriptsys.lua
index 515522d6443..7026410c5e4 100644
--- a/data/libs/functions/revscriptsys.lua
+++ b/data/libs/functions/revscriptsys.lua
@@ -290,6 +290,14 @@ do
self:eventType(MONSTERS_EVENT_SAY)
self:onSay(value)
return
+ elseif key == "onPlayerAttack" then
+ self:eventType(MONSTERS_EVENT_ATTACKED_BY_PLAYER)
+ self:onPlayerAttack(value)
+ return
+ elseif key == "onSpawn" then
+ self:eventType(MONSTERS_EVENT_ON_SPAWN)
+ self:onSpawn(value)
+ return
end
rawset(self, key, value)
end
diff --git a/data/libs/systems/zones.lua b/data/libs/systems/zones.lua
index ff37af5b16a..698a464fe87 100644
--- a/data/libs/systems/zones.lua
+++ b/data/libs/systems/zones.lua
@@ -100,7 +100,7 @@ setmetatable(ZoneEvent, {
function ZoneEvent:register()
if self.beforeEnter then
- local beforeEnter = EventCallback()
+ local beforeEnter = EventCallback("ZoneEventBeforeEnter", true)
function beforeEnter.zoneBeforeCreatureEnter(zone, creature)
if zone ~= self.zone then
return true
@@ -112,7 +112,7 @@ function ZoneEvent:register()
end
if self.beforeLeave then
- local beforeLeave = EventCallback()
+ local beforeLeave = EventCallback("ZoneEventBeforeLeave", true)
function beforeLeave.zoneBeforeCreatureLeave(zone, creature)
if zone ~= self.zone then
return true
@@ -124,7 +124,7 @@ function ZoneEvent:register()
end
if self.afterEnter then
- local afterEnter = EventCallback()
+ local afterEnter = EventCallback("ZoneEventAfterEnter", true)
function afterEnter.zoneAfterCreatureEnter(zone, creature)
if zone ~= self.zone then
return true
@@ -136,7 +136,7 @@ function ZoneEvent:register()
end
if self.afterLeave then
- local afterLeave = EventCallback()
+ local afterLeave = EventCallback("ZoneEventAfterLeave", true)
function afterLeave.zoneAfterCreatureLeave(zone, creature)
if zone ~= self.zone then
return true
@@ -148,7 +148,7 @@ function ZoneEvent:register()
end
if self.onSpawn then
- local afterEnter = EventCallback()
+ local afterEnter = EventCallback("ZoneEventAfterEnterOnSpawn", true)
function afterEnter.zoneAfterCreatureEnter(zone, creature)
if zone ~= self.zone then
return true
diff --git a/data/scripts/actions/items/cobra_flask.lua b/data/scripts/actions/items/cobra_flask.lua
index 095a8c39a96..c36d42983c5 100644
--- a/data/scripts/actions/items/cobra_flask.lua
+++ b/data/scripts/actions/items/cobra_flask.lua
@@ -1,4 +1,4 @@
-local applyCobraFlaskEffectOnMonsterSpawn = EventCallback()
+local applyCobraFlaskEffectOnMonsterSpawn = EventCallback("CobraFlaskEffectOnMonsterSpawn")
applyCobraFlaskEffectOnMonsterSpawn.monsterOnSpawn = function(monster, position)
if table.contains({ "cobra scout", "cobra vizier", "cobra assassin" }, monster:getName():lower()) then
diff --git a/data-otservbr-global/scripts/actions/tools/sickle.lua b/data/scripts/actions/tools/sickle.lua
similarity index 100%
rename from data-otservbr-global/scripts/actions/tools/sickle.lua
rename to data/scripts/actions/tools/sickle.lua
diff --git a/data/scripts/eventcallbacks/README.md b/data/scripts/eventcallbacks/README.md
index ae5de046bd2..6e0bacfcd6e 100644
--- a/data/scripts/eventcallbacks/README.md
+++ b/data/scripts/eventcallbacks/README.md
@@ -18,6 +18,7 @@ Event callbacks are available for several categories of game entities, such as `
- `(ReturnValue)` `creatureOnTargetCombat`
- `(void)` `creatureOnHear`
- `(void)` `creatureOnDrainHealth`
+- `(void)` `creatureOnCombat`
- `(bool)` `partyOnJoin`
- `(bool)` `partyOnLeave`
- `(bool)` `partyOnDisband`
@@ -62,7 +63,7 @@ Below are examples for each category of game entities:
### Creature Callback
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.creatureOnAreaCombat(creature, tile, isAggressive)
-- custom behavior when a creature enters combat area
@@ -75,7 +76,7 @@ callback:register()
### Player Callback
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.playerOnLook(player, position, thing, stackpos, lookDistance)
-- custom behavior when a player looks at something
@@ -87,7 +88,7 @@ callback:register()
### Party Callback
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.partyOnJoin(party, player)
-- custom behavior when a player joins a party
@@ -99,7 +100,7 @@ callback:register()
### Monster Callback
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.monsterOnSpawn(monster, position)
-- custom behavior when a monster spawns
@@ -111,7 +112,7 @@ callback:register()
### Npc Callback
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.npcOnSpawn(npc, position)
-- custom behavior when a npc spawns
@@ -129,7 +130,7 @@ If the callback returns `false`, the execution of the associated function on the
Here is an example of a boolean event callback:
```lua
-local callback = EventCallback()
+local callback = EventCallback("UniqueCallbackName")
function callback.playerOnMoveItem(player, item, count, fromPos, toPos, fromCylinder, toCylinder)
if item:getId() == ITEM_PARCEL then
@@ -180,7 +181,7 @@ Here is an example of defining multiple callbacks for the creatureOnAreaCombat e
#### Example 1
```lua
-local example1 = EventCallback()
+local example1 = EventCallback("UniqueCallbackName")
function example1.creatureOnAreaCombat(creature, tile, isAggressive)
-- custom behavior 1 when a creature enters combat area
@@ -192,7 +193,7 @@ example1:register()
#### Example 2
```lua
-local example2 = EventCallback()
+local example2 = EventCallback("UniqueCallbackName")
function example2.creatureOnAreaCombat(creature, tile, isAggressive)
-- custom behavior 2 when a creature enters combat area
diff --git a/data/scripts/eventcallbacks/creature/on_area_combat.lua b/data/scripts/eventcallbacks/creature/on_area_combat.lua
index f68cc95ccad..cd8720b25ad 100644
--- a/data/scripts/eventcallbacks/creature/on_area_combat.lua
+++ b/data/scripts/eventcallbacks/creature/on_area_combat.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("CreatureOnAreaCombatBaseEvent")
function callback.creatureOnAreaCombat(creature, tile, isAggressive)
return RETURNVALUE_NOERROR
diff --git a/data/scripts/eventcallbacks/creature/on_hear.lua b/data/scripts/eventcallbacks/creature/on_hear.lua
index 871f6456c9a..2954c81c8fb 100644
--- a/data/scripts/eventcallbacks/creature/on_hear.lua
+++ b/data/scripts/eventcallbacks/creature/on_hear.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("CreatureOnHearBaseEvent")
function callback.creatureOnHear(creature, speaker, words, type) end
diff --git a/data/scripts/eventcallbacks/monster/on_spawn.lua b/data/scripts/eventcallbacks/monster/on_spawn.lua
index e40778aafce..5a490a8edac 100644
--- a/data/scripts/eventcallbacks/monster/on_spawn.lua
+++ b/data/scripts/eventcallbacks/monster/on_spawn.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnSpawnBase")
function callback.monsterOnSpawn(monster, position)
if not monster then
diff --git a/data/scripts/eventcallbacks/monster/ondroploot__base.lua b/data/scripts/eventcallbacks/monster/ondroploot__base.lua
index cb7bc4207e8..0f724be9f67 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot__base.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot__base.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootBaseEvent")
function Player:canReceiveLoot()
return self:getStamina() > 840
@@ -15,14 +15,14 @@ function callback.monsterOnDropLoot(monster, corpse)
end
local mType = monster:getType()
if not mType then
- logger.warning("monsterOnDropLoot: monster has no type")
+ logger.warn("monsterOnDropLoot: monster '{}' has no type", monster:getName())
return
end
local charm = player and player:getCharmMonsterType(CHARM_GUT)
local gut = charm and charm:raceId() == mType:raceId()
- local lootTable = mType:generateLootRoll({ factor = factor, gut = gut }, {})
+ local lootTable = mType:generateLootRoll({ factor = factor, gut = gut }, {}, player)
corpse:addLoot(lootTable)
for _, item in ipairs(lootTable) do
if item.gut then
diff --git a/data/scripts/eventcallbacks/monster/ondroploot_boosted.lua b/data/scripts/eventcallbacks/monster/ondroploot_boosted.lua
index 1d94033a572..3bb43256772 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot_boosted.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot_boosted.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootBoosted")
function callback.monsterOnDropLoot(monster, corpse)
if not monster or not corpse then
@@ -22,7 +22,7 @@ function callback.monsterOnDropLoot(monster, corpse)
local factor = 1.0
local msgSuffix = " (boosted loot)"
- corpse:addLoot(mType:generateLootRoll({ factor = factor, gut = false }, {}))
+ corpse:addLoot(mType:generateLootRoll({ factor = factor, gut = false }, {}, player))
local existingSuffix = corpse:getAttribute(ITEM_ATTRIBUTE_LOOTMESSAGE_SUFFIX) or ""
corpse:setAttribute(ITEM_ATTRIBUTE_LOOTMESSAGE_SUFFIX, existingSuffix .. msgSuffix)
diff --git a/data/scripts/eventcallbacks/monster/ondroploot_gem_atelier.lua b/data/scripts/eventcallbacks/monster/ondroploot_gem_atelier.lua
index d1fb6ad38b1..4305e60e086 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot_gem_atelier.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot_gem_atelier.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootGemAtelier")
function callback.monsterOnDropLoot(monster, corpse)
if not monster or not corpse then
diff --git a/data/scripts/eventcallbacks/monster/ondroploot_hazard.lua b/data/scripts/eventcallbacks/monster/ondroploot_hazard.lua
index ddcaeaee782..78851d186e6 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot_hazard.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot_hazard.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootHazard")
function callback.monsterOnDropLoot(monster, corpse)
if not monster:hazard() then
@@ -31,7 +31,7 @@ function callback.monsterOnDropLoot(monster, corpse)
local lootTable = {}
for _ = 1, rolls do
- lootTable = mType:generateLootRoll({ factor = factor, gut = false }, lootTable)
+ lootTable = mType:generateLootRoll({ factor = factor, gut = false }, lootTable, player)
end
corpse:addLoot(lootTable)
diff --git a/data/scripts/eventcallbacks/monster/ondroploot_prey.lua b/data/scripts/eventcallbacks/monster/ondroploot_prey.lua
index 1f732f173d2..eb4657ccc4f 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot_prey.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot_prey.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootPrey")
function callback.monsterOnDropLoot(monster, corpse)
local player = Player(corpse:getCorpseOwner())
@@ -42,7 +42,7 @@ function callback.monsterOnDropLoot(monster, corpse)
msgSuffix = msgSuffix .. " (active prey bonus)"
end
- corpse:addLoot(mType:generateLootRoll({ factor = factor, gut = false }, {}))
+ corpse:addLoot(mType:generateLootRoll({ factor = factor, gut = false }, {}, player))
local existingSuffix = corpse:getAttribute(ITEM_ATTRIBUTE_LOOTMESSAGE_SUFFIX) or ""
corpse:setAttribute(ITEM_ATTRIBUTE_LOOTMESSAGE_SUFFIX, existingSuffix .. msgSuffix)
end
diff --git a/data/scripts/eventcallbacks/monster/ondroploot_wealth_duplex.lua b/data/scripts/eventcallbacks/monster/ondroploot_wealth_duplex.lua
index 20202c0b5cc..a4f3c67fe9c 100644
--- a/data/scripts/eventcallbacks/monster/ondroploot_wealth_duplex.lua
+++ b/data/scripts/eventcallbacks/monster/ondroploot_wealth_duplex.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterOnDropLootWealthDuplex")
function callback.monsterOnDropLoot(monster, corpse)
local player = Player(corpse:getCorpseOwner())
@@ -59,7 +59,7 @@ function callback.monsterOnDropLoot(monster, corpse)
local lootTable = {}
for _ = 1, rolls do
- lootTable = mType:generateLootRoll({ factor = factor, gut = false }, lootTable)
+ lootTable = mType:generateLootRoll({ factor = factor, gut = false }, lootTable, player)
end
corpse:addLoot(lootTable)
diff --git a/data/scripts/eventcallbacks/monster/postdroploot_analyzer.lua b/data/scripts/eventcallbacks/monster/postdroploot_analyzer.lua
index 8214ac98ea1..4c1e53e6f77 100644
--- a/data/scripts/eventcallbacks/monster/postdroploot_analyzer.lua
+++ b/data/scripts/eventcallbacks/monster/postdroploot_analyzer.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("MonsterPostDropLootAnalyzer")
function callback.monsterPostDropLoot(monster, corpse)
local player = Player(corpse:getCorpseOwner())
diff --git a/data/scripts/eventcallbacks/party/on_disband.lua b/data/scripts/eventcallbacks/party/on_disband.lua
index 60838962395..93bb1578fd1 100644
--- a/data/scripts/eventcallbacks/party/on_disband.lua
+++ b/data/scripts/eventcallbacks/party/on_disband.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PartyOnDisbandEventBaseEvent")
function callback.partyOnDisband(party)
local members = party:getMembers()
diff --git a/data/scripts/eventcallbacks/player/on_browse_field.lua b/data/scripts/eventcallbacks/player/on_browse_field.lua
index a4f00341202..3b674a608a7 100644
--- a/data/scripts/eventcallbacks/player/on_browse_field.lua
+++ b/data/scripts/eventcallbacks/player/on_browse_field.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnBrowseFieldBaseEvent")
function callback.playerOnBrowseField(player, position)
return true
diff --git a/data/scripts/eventcallbacks/player/on_look.lua b/data/scripts/eventcallbacks/player/on_look.lua
index 022aebbcc36..6b4be92553a 100644
--- a/data/scripts/eventcallbacks/player/on_look.lua
+++ b/data/scripts/eventcallbacks/player/on_look.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnLookBaseEvent")
function callback.playerOnLook(player, thing, position, distance)
local description = "You see "
diff --git a/data/scripts/eventcallbacks/player/on_look_in_shop.lua b/data/scripts/eventcallbacks/player/on_look_in_shop.lua
index bd96296624d..0e724009b5e 100644
--- a/data/scripts/eventcallbacks/player/on_look_in_shop.lua
+++ b/data/scripts/eventcallbacks/player/on_look_in_shop.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnLookInShopBaseEvent")
function callback.playerOnLookInShop(player, itemType, count)
return true
diff --git a/data/scripts/eventcallbacks/player/on_look_in_trade.lua b/data/scripts/eventcallbacks/player/on_look_in_trade.lua
index 77711c5b95d..c82dd9905f0 100644
--- a/data/scripts/eventcallbacks/player/on_look_in_trade.lua
+++ b/data/scripts/eventcallbacks/player/on_look_in_trade.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnLookInTradeBaseEvent")
function callback.playerOnLookInTrade(player, partner, item, distance)
player:sendTextMessage(MESSAGE_LOOK, "You see " .. item:getDescription(distance))
diff --git a/data/scripts/eventcallbacks/player/on_remove_count.lua b/data/scripts/eventcallbacks/player/on_remove_count.lua
index 4629d0a1b12..f39438c24ba 100644
--- a/data/scripts/eventcallbacks/player/on_remove_count.lua
+++ b/data/scripts/eventcallbacks/player/on_remove_count.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnRemoveCountBaseEvent")
function callback.playerOnRemoveCount(player, item)
player:sendWaste(item:getId())
diff --git a/data/scripts/eventcallbacks/player/on_request_quest_line.lua b/data/scripts/eventcallbacks/player/on_request_quest_line.lua
index c233279cfb8..ad5737d3246 100644
--- a/data/scripts/eventcallbacks/player/on_request_quest_line.lua
+++ b/data/scripts/eventcallbacks/player/on_request_quest_line.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnRequestQuestLineBaseEvent")
function callback.playerOnRequestQuestLine(player, questId)
player:sendQuestLine(questId)
diff --git a/data/scripts/eventcallbacks/player/on_request_quest_log.lua b/data/scripts/eventcallbacks/player/on_request_quest_log.lua
index b6bcbbc3a64..15cfbd34719 100644
--- a/data/scripts/eventcallbacks/player/on_request_quest_log.lua
+++ b/data/scripts/eventcallbacks/player/on_request_quest_log.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnRequestQuestLogBaseEvent")
function callback.playerOnRequestQuestLog(player)
player:sendQuestLog()
diff --git a/data/scripts/eventcallbacks/player/on_rotate_item.lua b/data/scripts/eventcallbacks/player/on_rotate_item.lua
index 3692fcdc655..61bb2a99b2a 100644
--- a/data/scripts/eventcallbacks/player/on_rotate_item.lua
+++ b/data/scripts/eventcallbacks/player/on_rotate_item.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnRotateItemBaseEvent")
function callback.playerOnRotateItem(player, item, position)
if item:getActionId() == IMMOVABLE_ACTION_ID then
diff --git a/data/scripts/eventcallbacks/player/on_storage_update.lua b/data/scripts/eventcallbacks/player/on_storage_update.lua
index 8b3006b1d54..0f4233e51d2 100644
--- a/data/scripts/eventcallbacks/player/on_storage_update.lua
+++ b/data/scripts/eventcallbacks/player/on_storage_update.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnStorageUpdateBaseEvent")
function callback.playerOnStorageUpdate(player, key, value, oldValue, currentFrameTime)
player:updateStorage(key, value, oldValue, currentFrameTime)
diff --git a/data/scripts/eventcallbacks/player/on_trade_accept.lua b/data/scripts/eventcallbacks/player/on_trade_accept.lua
index 3d1eba7171b..8425bdc10b3 100644
--- a/data/scripts/eventcallbacks/player/on_trade_accept.lua
+++ b/data/scripts/eventcallbacks/player/on_trade_accept.lua
@@ -1,4 +1,4 @@
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnTradeAcceptBaseEvent")
function callback.playerOnTradeAccept(player, target, item, targetItem)
player:closeForge()
diff --git a/data/scripts/lib/quests.lua b/data/scripts/lib/quests.lua
new file mode 100644
index 00000000000..08005e34ae2
--- /dev/null
+++ b/data/scripts/lib/quests.lua
@@ -0,0 +1,2 @@
+-- We need to register the variables beforehand to avoid accessing null values.
+RegisterSoulWarBossesLevers()
diff --git a/data/scripts/lib/register_lever_tables.lua b/data/scripts/lib/register_lever_tables.lua
index 66a78f42a11..de3cba6649e 100644
--- a/data/scripts/lib/register_lever_tables.lua
+++ b/data/scripts/lib/register_lever_tables.lua
@@ -7,8 +7,8 @@ AscendingFerumbrasConfig = {
centerRoom = Position(33392, 31473, 14), -- Center Room
exitPosition = Position(33266, 31479, 14), -- Exit Position
newPos = Position(33392, 31479, 14), -- Player Position on room
- days = 3,
+ days = 5,
range = 20,
time = 60, -- time in minutes to remove the player
- vortex = 23726,
+ vortex = 20121,
}
diff --git a/data/scripts/lib/register_monster_type.lua b/data/scripts/lib/register_monster_type.lua
index e42bf4e6ca1..cfb0a6bfaaa 100644
--- a/data/scripts/lib/register_monster_type.lua
+++ b/data/scripts/lib/register_monster_type.lua
@@ -940,8 +940,8 @@ function readSpell(incomingLua, mtype)
if incomingLua.effect then
spell:setCombatEffect(incomingLua.effect)
end
- if incomingLua.shootEffect then
- spell:setCombatShootEffect(incomingLua.shootEffect)
+ if incomingLua.shootEffect or incomingLua.shooteffect then
+ spell:setCombatShootEffect(incomingLua.shootEffect or incomingLua.shooteffect)
end
end
diff --git a/data/scripts/lib/register_npc_type.lua b/data/scripts/lib/register_npc_type.lua
index 53a5d525e59..e191bbbe240 100644
--- a/data/scripts/lib/register_npc_type.lua
+++ b/data/scripts/lib/register_npc_type.lua
@@ -141,25 +141,57 @@ registerNpcType.events = function(npcType, mask)
end
end
+-- Global item tracker to track buy and sell prices across all NPCs
+NpcPriceChecker = NpcPriceChecker or {}
+
registerNpcType.shop = function(npcType, mask)
if type(mask.shop) == "table" then
for _, shopItems in pairs(mask.shop) do
local parent = Shop()
- if shopItems.itemName or shopItems.itemname then
- parent:setNameItem(shopItems.itemName or shopItems.itemname)
+ local itemName = shopItems.itemName or shopItems.itemname
+ local clientId = shopItems.clientId or shopItems.clientid
+ local buyPrice = shopItems.buy
+ local sellPrice = shopItems.sell
+ local npcName = npcType:getName() -- Assuming `npcType` has a `getName` method to get the NPC's name
+
+ if itemName then
+ parent:setNameItem(itemName)
end
- if shopItems.clientId or shopItems.clientid then
- parent:setId(shopItems.clientId or shopItems.clientid)
+ if clientId then
+ parent:setId(clientId)
end
if shopItems.subType or shopItems.subtype or shopItems.count then
parent:setCount(shopItems.subType or shopItems.subtype or shopItems.count)
end
- if shopItems.buy then
- parent:setBuyPrice(shopItems.buy)
+ if buyPrice then
+ parent:setBuyPrice(buyPrice)
+ end
+ if sellPrice then
+ parent:setSellPrice(sellPrice)
end
- if shopItems.sell then
- parent:setSellPrice(shopItems.sell)
+
+ if clientId then
+ if not NpcPriceChecker[clientId] then
+ NpcPriceChecker[clientId] = { buy = nil, sell = nil, buyNpc = nil, sellNpc = nil }
+ end
+
+ if buyPrice then
+ NpcPriceChecker[clientId].buy = buyPrice
+ NpcPriceChecker[clientId].buyNpc = npcName
+ end
+
+ if sellPrice then
+ NpcPriceChecker[clientId].sell = sellPrice
+ NpcPriceChecker[clientId].sellNpc = npcName
+ end
+
+ if NpcPriceChecker[clientId].buy and NpcPriceChecker[clientId].sell then
+ if NpcPriceChecker[clientId].sell > NpcPriceChecker[clientId].buy then
+ logger.warn("The item {} ({}) is being sold for a value greater than the value it is purchased for by the NPCs. Buy NPC: {}, Sell NPC: {}", itemName, clientId, NpcPriceChecker[clientId].buyNpc, NpcPriceChecker[clientId].sellNpc)
+ end
+ end
end
+
if shopItems.storageKey or shopItems.storagekey then
parent:setStorageKey(shopItems.storageKey or shopItems.storagekey)
end
@@ -169,26 +201,26 @@ registerNpcType.shop = function(npcType, mask)
if shopItems.child then
for _, children in pairs(shopItems.child) do
local child = Shop()
- if shopItems.itemName or shopItems.itemname then
- child:setNameItem(shopItems.itemName or shopItems.itemname)
+ if children.itemName or children.itemname then
+ child:setNameItem(children.itemName or children.itemname)
end
- if shopItems.clientId or shopItems.clientid then
- child:setId(shopItems.clientId or shopItems.clientid)
+ if children.clientId or children.clientid then
+ child:setId(children.clientId or children.clientid)
end
- if shopItems.subType or shopItems.subtype or shopItems.count then
- child:setCount(shopItems.subType or shopItems.subtype or shopItems.count)
+ if children.subType or children.subtype or children.count then
+ child:setCount(children.subType or children.subtype or children.count)
end
- if shopItems.buy then
- child:setBuyPrice(shopItems.buy)
+ if children.buy then
+ child:setBuyPrice(children.buy)
end
- if shopItems.sell then
- child:setSellPrice(shopItems.sell)
+ if children.sell then
+ child:setSellPrice(children.sell)
end
- if shopItems.storageKey or shopItems.storagekey then
- child:setStorageKey(shopItems.storageKey or shopItems.storagekey)
+ if children.storageKey or children.storagekey then
+ child:setStorageKey(children.storageKey or children.storagekey)
end
- if shopItems.storageValue or shopItems.storagevalue then
- child:setStorageValue(shopItems.storageValue or shopItems.storagevalue)
+ if children.storageValue or children.storagevalue then
+ child:setStorageValue(children.storageValue or children.storagevalue)
end
parent:addChildShop(child)
end
diff --git a/data/scripts/lib/register_spells.lua b/data/scripts/lib/register_spells.lua
index 49a5d7aec2f..8c549859640 100644
--- a/data/scripts/lib/register_spells.lua
+++ b/data/scripts/lib/register_spells.lua
@@ -393,6 +393,12 @@ AREA_RING1_BURST3 = {
{ 0, 0, 0, 1, 1, 1, 0, 0, 0 },
}
+CrossBeamArea3X2 = {
+ { 1, 1, 1 },
+ { 0, 1, 0 },
+ { 0, 3, 0 },
+}
+
-- The numbered-keys represents the damage values, and their table
-- contains the minimum and maximum number of rounds of those damage values.
RANGE = {
diff --git a/data/scripts/spells/healing/heal_malice.lua b/data/scripts/spells/healing/heal_malice.lua
new file mode 100644
index 00000000000..67211a58341
--- /dev/null
+++ b/data/scripts/spells/healing/heal_malice.lua
@@ -0,0 +1,29 @@
+function onTargetCreature(creature, target)
+ if target:getName() == "Goshnar's Malice" then
+ logger.debug("Monster {} Healing {}", creature:getName(), target:getName())
+ local min = 15000
+ local max = 30000
+ doTargetCombatHealth(target, target, COMBAT_HEALING, min, max)
+ end
+ return true
+end
+
+local combat = Combat()
+combat:setParameter(COMBAT_PARAM_EFFECT, CONST_ME_MAGIC_RED)
+combat:setParameter(COMBAT_PARAM_AGGRESSIVE, 0)
+combat:setParameter(COMBAT_PARAM_DISPEL, CONDITION_PARALYZE)
+combat:setArea(createCombatArea(AREA_CIRCLE3X3))
+combat:setCallback(CALLBACK_PARAM_TARGETCREATURE, "onTargetCreature")
+
+local spell = Spell("instant")
+
+function spell.onCastSpell(creature, var)
+ return combat:execute(creature, var)
+end
+
+spell:name("Heal Malice")
+spell:words("#####462")
+spell:blockWalls(true)
+spell:needLearn(true)
+spell:needDirection(true)
+spell:register()
diff --git a/data/scripts/systems/reward_chest.lua b/data/scripts/systems/reward_chest.lua
index 384115af0b0..07ec9b512a9 100644
--- a/data/scripts/systems/reward_chest.lua
+++ b/data/scripts/systems/reward_chest.lua
@@ -103,9 +103,9 @@ function bossDeath.onDeath(creature, corpse, killer, mostDamageKiller, lastHitUn
end
local playerLoot = creature:generateGemAtelierLoot()
- playerLoot = monsterType:getBossReward(lootFactor, _ == 1, false, playerLoot)
+ playerLoot = monsterType:getBossReward(lootFactor, _ == 1, false, playerLoot, player)
for _ = 2, rolls do
- playerLoot = monsterType:getBossReward(lootFactor, false, true, playerLoot)
+ playerLoot = monsterType:getBossReward(lootFactor, false, true, playerLoot, player)
end
-- Add droped items to reward container
diff --git a/data/scripts/talkactions/gm/afk.lua b/data/scripts/talkactions/gm/afk.lua
index 6167a2b6068..f5342362113 100644
--- a/data/scripts/talkactions/gm/afk.lua
+++ b/data/scripts/talkactions/gm/afk.lua
@@ -69,7 +69,7 @@ afkEffect:interval(5000)
afkEffect:register()
------------------ Stop AFK Message when moves ------------------
-local callback = EventCallback()
+local callback = EventCallback("PlayerOnWalk")
function callback.playerOnWalk(player, creature, creaturePos, toPos)
local isAfk = checkIsAFK(player:getId())
if isAfk.afk then
diff --git a/data/scripts/talkactions/gm/distance_effect.lua b/data/scripts/talkactions/gm/distance_effect.lua
new file mode 100644
index 00000000000..4c637972cb7
--- /dev/null
+++ b/data/scripts/talkactions/gm/distance_effect.lua
@@ -0,0 +1,37 @@
+local magicEffect = TalkAction("/distanceeffect")
+
+function magicEffect.onSay(player, words, param)
+ -- create log
+ logCommand(player, words, param)
+
+ if param == "" then
+ player:sendCancelMessage("Command param required.")
+ return true
+ end
+
+ local effect = tonumber(param)
+ if effect ~= nil and effect > 0 then
+ local playerPos = player:getPosition()
+ local direction = player:getDirection()
+ local targetPos = Position(playerPos.x, playerPos.y, playerPos.z)
+
+ local distance = 7
+ if direction == DIRECTION_NORTH then
+ targetPos.y = targetPos.y - distance
+ elseif direction == DIRECTION_EAST then
+ targetPos.x = targetPos.x + distance
+ elseif direction == DIRECTION_SOUTH then
+ targetPos.y = targetPos.y + distance
+ elseif direction == DIRECTION_WEST then
+ targetPos.x = targetPos.x - distance
+ end
+
+ player:getPosition():sendDistanceEffect(targetPos, effect)
+ end
+
+ return true
+end
+
+magicEffect:separator(" ")
+magicEffect:groupType("gamemaster")
+magicEffect:register()
diff --git a/data/scripts/talkactions/gm/position.lua b/data/scripts/talkactions/gm/position.lua
index 1869b109c5f..dc95552d5c2 100644
--- a/data/scripts/talkactions/gm/position.lua
+++ b/data/scripts/talkactions/gm/position.lua
@@ -1,23 +1,5 @@
local position = TalkAction("/pos", "!pos")
-local function extractCoordinates(input)
- local patterns = {
- -- table format
- "{%s*x%s*=%s*(%d+)%s*,%s*y%s*=%s*(%d+)%s*,%s*z%s*=%s*(%d+)%s*}",
- -- Position format
- "Position%s*%((%d+)%s*,%s*(%d+)%s*,%s*(%d+)%s*%)",
- -- x, y, z format
- "(%d+)%s*,%s*(%d+)%s*,%s*(%d+)",
- }
-
- for _, pattern in ipairs(patterns) do
- local x, y, z = string.match(input, pattern)
- if x and y and z then
- return tonumber(x), tonumber(y), tonumber(z)
- end
- end
-end
-
function position.onSay(player, words, param)
-- create log
logCommand(player, words, param)
@@ -28,18 +10,21 @@ function position.onSay(player, words, param)
return
end
- local x, y, z = extractCoordinates(param)
- if x and y and z then
- local teleportPosition = Position(x, y, z)
- local tile = Tile(teleportPosition)
- if not tile then
- player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Invalid tile or position. Send a valid position.")
- return
- end
-
- player:teleportTo(teleportPosition)
- else
+ local teleportPosition = param:toPosition()
+ if not teleportPosition then
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Invalid position format. Use one of the following formats: \n/pos {x = ..., y = ..., z = ...}\n/pos Position(..., ..., ...)\n/pos x, y, z.")
+ return
+ end
+
+ local tile = Tile(teleportPosition)
+ if not tile then
+ player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Invalid tile or position. Send a valid position.")
+ return
+ end
+
+ player:teleportTo(teleportPosition)
+ if not player:isInGhostMode() then
+ teleportPosition:sendMagicEffect(CONST_ME_TELEPORT)
end
end
diff --git a/data/scripts/talkactions/god/add_condition.lua b/data/scripts/talkactions/god/add_condition.lua
new file mode 100644
index 00000000000..7d92cbc0a09
--- /dev/null
+++ b/data/scripts/talkactions/god/add_condition.lua
@@ -0,0 +1,10 @@
+local talkaction = TalkAction("/testtaintconditions")
+
+function talkaction.onSay(player, words, param)
+ player:setTaintIcon()
+ return false
+end
+
+talkaction:separator(" ")
+talkaction:groupType("god")
+talkaction:register()
diff --git a/data/scripts/talkactions/god/icons_functions.lua b/data/scripts/talkactions/god/icons_functions.lua
index 9f1ee85c0f3..059d7ac6ed9 100644
--- a/data/scripts/talkactions/god/icons_functions.lua
+++ b/data/scripts/talkactions/god/icons_functions.lua
@@ -78,11 +78,7 @@ function bakragoreIcon.onSay(player, words, param)
end
if param == "remove" then
- for i = 1, 10 do
- if player:hasCondition(CONDITION_BAKRAGORE, i) then
- player:removeCondition(CONDITION_BAKRAGORE, i)
- end
- end
+ player:removeIconBakragore()
player:sendTextMessage(MESSAGE_EVENT_ADVANCE, "Removed all Bakragore icons.")
return true
end
diff --git a/src/canary_server.cpp b/src/canary_server.cpp
index e49a86d7d9f..cca280b69fa 100644
--- a/src/canary_server.cpp
+++ b/src/canary_server.cpp
@@ -367,6 +367,7 @@ void CanaryServer::loadModules() {
modulesLoadHelper(g_modules().loadFromXml(), "modules/modules.xml");
logger.debug("Loading datapack scripts on folder: {}/", datapackName);
+ modulesLoadHelper(g_scripts().loadScripts(datapackFolder + "/scripts/lib", true, false), datapackFolder + "/scripts/libs");
// Load scripts
modulesLoadHelper(g_scripts().loadScripts(datapackFolder + "/scripts", false, false), datapackFolder + "/scripts");
// Load monsters
diff --git a/src/config/configmanager.cpp b/src/config/configmanager.cpp
index 6390d0699b3..80126ddac93 100644
--- a/src/config/configmanager.cpp
+++ b/src/config/configmanager.cpp
@@ -379,7 +379,7 @@ bool ConfigManager::reload() {
}
void ConfigManager::missingConfigWarning(const char* identifier) {
- g_logger().warn("[{}]: Missing configuration for identifier: {}", __FUNCTION__, identifier);
+ g_logger().debug("[{}]: Missing configuration for identifier: {}", __FUNCTION__, identifier);
}
std::string ConfigManager::loadStringConfig(lua_State* L, const ConfigKey_t &key, const char* identifier, const std::string &defaultValue) {
diff --git a/src/creatures/appearance/outfit/outfit.hpp b/src/creatures/appearance/outfit/outfit.hpp
index 0d89a2c932d..c4d49a7f1a2 100644
--- a/src/creatures/appearance/outfit/outfit.hpp
+++ b/src/creatures/appearance/outfit/outfit.hpp
@@ -54,6 +54,16 @@ class Outfits {
return outfits[sex];
}
+ std::shared_ptr getOutfitByName(PlayerSex_t sex, const std::string &name) const {
+ for (const auto &outfit : outfits[sex]) {
+ if (outfit->name == name) {
+ return outfit;
+ }
+ }
+
+ return nullptr;
+ }
+
private:
std::vector> outfits[PLAYERSEX_LAST + 1];
};
diff --git a/src/creatures/combat/combat.cpp b/src/creatures/combat/combat.cpp
index 8e3de0da1bd..d65d10e4e6f 100644
--- a/src/creatures/combat/combat.cpp
+++ b/src/creatures/combat/combat.cpp
@@ -12,6 +12,8 @@
#include "declarations.hpp"
#include "creatures/combat/combat.hpp"
#include "lua/creature/events.hpp"
+#include "lua/callbacks/event_callback.hpp"
+#include "lua/callbacks/events_callbacks.hpp"
#include "creatures/players/wheel/player_wheel.hpp"
#include "game/game.hpp"
#include "game/scheduling/dispatcher.hpp"
@@ -595,10 +597,15 @@ void Combat::CombatHealthFunc(std::shared_ptr caster, std::shared_ptr<
targetPlayer = target->getPlayer();
}
+ g_logger().trace("[{}] (old) eventcallback: 'creatureOnCombat', damage primary: '{}', secondary: '{}'", __FUNCTION__, damage.primary.value, damage.secondary.value);
+ g_callbacks().executeCallback(EventCallback_t::creatureOnCombat, &EventCallback::creatureOnCombat, caster, target, std::ref(damage));
+ g_logger().trace("[{}] (new) eventcallback: 'creatureOnCombat', damage primary: '{}', secondary: '{}'", __FUNCTION__, damage.primary.value, damage.secondary.value);
+
if (attackerPlayer) {
std::shared_ptr- item = attackerPlayer->getWeapon();
damage = applyImbuementElementalDamage(attackerPlayer, item, damage);
g_events().eventPlayerOnCombat(attackerPlayer, target, item, damage);
+ g_callbacks().executeCallback(EventCallback_t::playerOnCombat, &EventCallback::playerOnCombat, attackerPlayer, target, item, std::ref(damage));
if (targetPlayer && targetPlayer->getSkull() != SKULL_BLACK) {
if (damage.primary.type != COMBAT_HEALING) {
@@ -624,6 +631,9 @@ void Combat::CombatHealthFunc(std::shared_ptr caster, std::shared_ptr<
damage.primary.value += static_cast(std::ceil((damage.primary.value * slot->bonusPercentage) / 100));
damage.secondary.value += static_cast(std::ceil((damage.secondary.value * slot->bonusPercentage) / 100));
}
+
+ // Monster type onPlayerAttack event
+ targetMonster->onAttackedByPlayer(attackerPlayer);
}
// Monster attacking player
@@ -1157,7 +1167,9 @@ void Combat::CombatFunc(std::shared_ptr caster, const Position &origin
if (CreatureVector* creatures = tile->getCreatures()) {
const std::shared_ptr topCreature = tile->getTopCreature();
- for (auto &creature : *creatures) {
+ // A copy of the tile's creature list is made because modifications to this vector, such as adding or removing creatures through a Lua callback, may occur during the iteration within the for loop.
+ CreatureVector creaturesCopy = *creatures;
+ for (auto &creature : creaturesCopy) {
if (params.targetCasterOrTopMost) {
if (caster && caster->getTile() == tile) {
if (creature != caster) {
@@ -1212,7 +1224,9 @@ void Combat::CombatFunc(std::shared_ptr caster, const Position &origin
if (CreatureVector* creatures = tile->getCreatures()) {
const std::shared_ptr topCreature = tile->getTopCreature();
- for (auto &creature : *creatures) {
+ // A copy of the tile's creature list is made because modifications to this vector, such as adding or removing creatures through a Lua callback, may occur during the iteration within the for loop.
+ CreatureVector creaturesCopy = *creatures;
+ for (auto &creature : creaturesCopy) {
if (params.targetCasterOrTopMost) {
if (caster && caster->getTile() == tile) {
if (creature != caster) {
diff --git a/src/creatures/combat/condition.cpp b/src/creatures/combat/condition.cpp
index c06d9a792d5..6784e8786f3 100644
--- a/src/creatures/combat/condition.cpp
+++ b/src/creatures/combat/condition.cpp
@@ -257,6 +257,8 @@ std::shared_ptr Condition::createCondition(ConditionId_t id, Conditio
return std::make_shared(id, type, ticks, buff, subId);
case CONDITION_BAKRAGORE:
return std::make_shared(id, type, ticks, buff, subId, isPersistent);
+ case CONDITION_GOSHNARTAINT:
+ return std::make_shared(id, type, ticks, buff, subId);
default:
return nullptr;
@@ -345,7 +347,14 @@ bool Condition::isRemovableOnDeath() const {
return false;
}
- if (conditionType == CONDITION_SPELLCOOLDOWN || conditionType == CONDITION_SPELLGROUPCOOLDOWN || conditionType == CONDITION_MUTED) {
+ static const std::unordered_set nonRemovableConditions = {
+ CONDITION_SPELLCOOLDOWN,
+ CONDITION_SPELLGROUPCOOLDOWN,
+ CONDITION_MUTED,
+ CONDITION_GOSHNARTAINT
+ };
+
+ if (nonRemovableConditions.find(conditionType) != nonRemovableConditions.end()) {
return false;
}
@@ -418,7 +427,26 @@ std::unordered_set ConditionGeneric::getIcons() const {
case CONDITION_ROOTED:
icons.insert(PlayerIcon::Rooted);
break;
-
+ case CONDITION_GOSHNARTAINT:
+ switch (subId) {
+ case 1:
+ icons.insert(PlayerIcon::GoshnarTaint1);
+ break;
+ case 2:
+ icons.insert(PlayerIcon::GoshnarTaint2);
+ break;
+ case 3:
+ icons.insert(PlayerIcon::GoshnarTaint3);
+ break;
+ case 4:
+ icons.insert(PlayerIcon::GoshnarTaint4);
+ break;
+ case 5:
+ icons.insert(PlayerIcon::GoshnarTaint5);
+ break;
+ default:
+ break;
+ }
default:
break;
}
diff --git a/src/creatures/creature.cpp b/src/creatures/creature.cpp
index 6c15f6cf1fe..a5cd66a629f 100644
--- a/src/creatures/creature.cpp
+++ b/src/creatures/creature.cpp
@@ -1804,7 +1804,7 @@ void Creature::handleLostSummon(bool teleportSummons) {
g_game().addMagicEffect(getPosition(), CONST_ME_POFF);
}
-int32_t Creature::getReflectPercent(CombatType_t combatType, bool useCharges /*= false*/) const {
+double_t Creature::getReflectPercent(CombatType_t combatType, bool useCharges /*= false*/) const {
try {
return reflectPercent.at(combatTypeToIndex(combatType));
} catch (const std::out_of_range &e) {
diff --git a/src/creatures/creature.hpp b/src/creatures/creature.hpp
index 9386716c1fd..08eac25824e 100644
--- a/src/creatures/creature.hpp
+++ b/src/creatures/creature.hpp
@@ -255,6 +255,10 @@ class Creature : virtual public Thing, public SharedObject {
return creatureIcons.at(key);
}
+ bool hasIcon(const std::string &key) const {
+ return creatureIcons.contains(key);
+ }
+
void setIcon(const std::string &key, CreatureIcon icon) {
creatureIcons[key] = icon;
iconChanged();
@@ -609,7 +613,7 @@ class Creature : virtual public Thing, public SharedObject {
* @param useCharges Indicates whether charges should be considered.
* @return The reflection percentage for the specified combat type.
*/
- virtual int32_t getReflectPercent(CombatType_t combatType, bool useCharges = false) const;
+ virtual double_t getReflectPercent(CombatType_t combatType, bool useCharges = false) const;
/**
* @brief Retrieves the flat reflection value for a given combat type.
@@ -707,6 +711,10 @@ class Creature : virtual public Thing, public SharedObject {
return false;
}
+ virtual bool isDead() const {
+ return false;
+ }
+
static constexpr int32_t mapWalkWidth = MAP_MAX_VIEW_PORT_X * 2 + 1;
static constexpr int32_t mapWalkHeight = MAP_MAX_VIEW_PORT_Y * 2 + 1;
static constexpr int32_t maxWalkCacheWidth = (mapWalkWidth - 1) / 2;
diff --git a/src/creatures/creatures_definitions.hpp b/src/creatures/creatures_definitions.hpp
index f1b2be8fdc6..3d975968ff0 100644
--- a/src/creatures/creatures_definitions.hpp
+++ b/src/creatures/creatures_definitions.hpp
@@ -113,15 +113,11 @@ enum ConditionType_t : uint8_t {
CONDITION_LESSERHEX = 31,
CONDITION_INTENSEHEX = 32,
CONDITION_GREATERHEX = 33,
- CONDITION_GOSHNAR1 = 34,
- CONDITION_GOSHNAR2 = 35,
- CONDITION_GOSHNAR3 = 36,
- CONDITION_GOSHNAR4 = 37,
- CONDITION_GOSHNAR5 = 38,
- CONDITION_BAKRAGORE = 39,
+ CONDITION_BAKRAGORE = 34,
+ CONDITION_GOSHNARTAINT = 35,
// Need the last ever
- CONDITION_COUNT = 39
+ CONDITION_COUNT
};
// constexpr definiting suppressible conditions
@@ -492,12 +488,14 @@ enum BestiaryType_t : uint8_t {
};
enum MonstersEvent_t : uint8_t {
- MONSTERS_EVENT_NONE = 0,
- MONSTERS_EVENT_THINK = 1,
- MONSTERS_EVENT_APPEAR = 2,
- MONSTERS_EVENT_DISAPPEAR = 3,
- MONSTERS_EVENT_MOVE = 4,
- MONSTERS_EVENT_SAY = 5,
+ MONSTERS_EVENT_NONE,
+ MONSTERS_EVENT_THINK,
+ MONSTERS_EVENT_APPEAR,
+ MONSTERS_EVENT_DISAPPEAR,
+ MONSTERS_EVENT_MOVE,
+ MONSTERS_EVENT_SAY,
+ MONSTERS_EVENT_ATTACKED_BY_PLAYER,
+ MONSTERS_EVENT_ON_SPAWN,
};
enum NpcsEvent_t : uint8_t {
diff --git a/src/creatures/monsters/monster.cpp b/src/creatures/monsters/monster.cpp
index 3ae73f8dc7c..1a364ece2ba 100644
--- a/src/creatures/monsters/monster.cpp
+++ b/src/creatures/monsters/monster.cpp
@@ -108,15 +108,57 @@ bool Monster::canWalkOnFieldType(CombatType_t combatType) const {
}
}
-int32_t Monster::getReflectPercent(CombatType_t reflectType, bool useCharges) const {
- int32_t result = Creature::getReflectPercent(reflectType, useCharges);
+double_t Monster::getReflectPercent(CombatType_t reflectType, bool useCharges) const {
+ // Monster type reflect
+ auto result = Creature::getReflectPercent(reflectType, useCharges);
+ if (result != 0) {
+ g_logger().debug("[{}] before mtype reflect element {}, percent {}", __FUNCTION__, fmt::underlying(reflectType), result);
+ }
auto it = mType->info.reflectMap.find(reflectType);
if (it != mType->info.reflectMap.end()) {
result += it->second;
}
+
+ if (result != 0) {
+ g_logger().debug("[{}] after mtype reflect element {}, percent {}", __FUNCTION__, fmt::underlying(reflectType), result);
+ }
+
+ // Monster reflect
+ auto monsterReflectIt = m_reflectElementMap.find(reflectType);
+ if (monsterReflectIt != m_reflectElementMap.end()) {
+ result += monsterReflectIt->second;
+ }
+
+ if (result != 0) {
+ g_logger().debug("[{}] (final) after monster reflect element {}, percent {}", __FUNCTION__, fmt::underlying(reflectType), result);
+ }
+
return result;
}
+void Monster::addReflectElement(CombatType_t combatType, int32_t percent) {
+ g_logger().debug("[{}] added reflect element {}, percent {}", __FUNCTION__, fmt::underlying(combatType), percent);
+ m_reflectElementMap[combatType] += percent;
+}
+
+int32_t Monster::getDefense() const {
+ auto mtypeDefense = mType->info.defense;
+ if (mtypeDefense != 0) {
+ g_logger().trace("[{}] old defense {}", __FUNCTION__, mtypeDefense);
+ }
+ mtypeDefense += m_defense;
+ if (mtypeDefense != 0) {
+ g_logger().trace("[{}] new defense {}", __FUNCTION__, mtypeDefense);
+ }
+ return mtypeDefense * getDefenseMultiplier();
+}
+
+void Monster::addDefense(int32_t defense) {
+ g_logger().trace("[{}] adding defense {}", __FUNCTION__, defense);
+ m_defense += defense;
+ g_logger().trace("[{}] new defense {}", __FUNCTION__, m_defense);
+}
+
uint32_t Monster::getHealingCombatValue(CombatType_t healingType) const {
auto it = mType->info.healingMap.find(healingType);
if (it != mType->info.healingMap.end()) {
@@ -315,6 +357,57 @@ void Monster::onCreatureSay(std::shared_ptr creature, SpeakClasses typ
}
}
+void Monster::onAttackedByPlayer(std::shared_ptr attackerPlayer) {
+ if (mType->info.monsterAttackedByPlayerEvent != -1) {
+ // onPlayerAttack(self, attackerPlayer)
+ LuaScriptInterface* scriptInterface = mType->info.scriptInterface;
+ if (!scriptInterface->reserveScriptEnv()) {
+ g_logger().error("Monster {} creature {}] Call stack overflow. Too many lua "
+ "script calls being nested.",
+ getName(), this->getName());
+ return;
+ }
+
+ ScriptEnvironment* env = scriptInterface->getScriptEnv();
+ env->setScriptId(mType->info.monsterAttackedByPlayerEvent, scriptInterface);
+
+ lua_State* L = scriptInterface->getLuaState();
+ scriptInterface->pushFunction(mType->info.monsterAttackedByPlayerEvent);
+
+ LuaScriptInterface::pushUserdata(L, getMonster());
+ LuaScriptInterface::setMetatable(L, -1, "Monster");
+
+ LuaScriptInterface::pushUserdata(L, attackerPlayer);
+ LuaScriptInterface::setMetatable(L, -1, "Player");
+
+ scriptInterface->callVoidFunction(2);
+ }
+}
+
+void Monster::onSpawn() {
+ if (mType->info.spawnEvent != -1) {
+ // onSpawn(self)
+ LuaScriptInterface* scriptInterface = mType->info.scriptInterface;
+ if (!scriptInterface->reserveScriptEnv()) {
+ g_logger().error("Monster {} creature {}] Call stack overflow. Too many lua "
+ "script calls being nested.",
+ getName(), this->getName());
+ return;
+ }
+
+ ScriptEnvironment* env = scriptInterface->getScriptEnv();
+ env->setScriptId(mType->info.spawnEvent, scriptInterface);
+
+ lua_State* L = scriptInterface->getLuaState();
+ scriptInterface->pushFunction(mType->info.spawnEvent);
+
+ LuaScriptInterface::pushUserdata(L, getMonster());
+ LuaScriptInterface::setMetatable(L, -1, "Monster");
+
+ scriptInterface->callVoidFunction(1);
+ }
+}
+
void Monster::addFriend(const std::shared_ptr &creature) {
if (creature == getMonster()) {
g_logger().error("[{}]: adding creature is same of monster", __FUNCTION__);
@@ -1921,6 +2014,8 @@ void Monster::death(std::shared_ptr) {
if (mType) {
g_game().sendSingleSoundEffect(static_self_cast()->getPosition(), mType->info.deathSound, getMonster());
}
+
+ setDead(true);
}
std::shared_ptr
- Monster::getCorpse(std::shared_ptr lastHitCreature, std::shared_ptr mostDamageCreature) {
@@ -2122,11 +2217,11 @@ bool Monster::changeTargetDistance(int32_t distance, uint32_t duration /* = 1200
}
bool Monster::isImmune(ConditionType_t conditionType) const {
- return mType->info.m_conditionImmunities[static_cast(conditionType)];
+ return m_isImmune || mType->info.m_conditionImmunities[static_cast(conditionType)];
}
bool Monster::isImmune(CombatType_t combatType) const {
- return mType->info.m_damageImmunities[combatTypeToIndex(combatType)];
+ return m_isImmune || mType->info.m_damageImmunities[combatTypeToIndex(combatType)];
}
void Monster::getPathSearchParams(const std::shared_ptr &creature, FindPathParams &fpp) {
diff --git a/src/creatures/monsters/monster.hpp b/src/creatures/monsters/monster.hpp
index 061ad2b9879..155ba1e5c87 100644
--- a/src/creatures/monsters/monster.hpp
+++ b/src/creatures/monsters/monster.hpp
@@ -77,9 +77,9 @@ class Monster final : public Creature {
int32_t getArmor() const override {
return mType->info.armor * getDefenseMultiplier();
}
- int32_t getDefense() const override {
- return mType->info.defense * getDefenseMultiplier();
- }
+ int32_t getDefense() const override;
+
+ void addDefense(int32_t defense);
Faction_t getFaction() const override {
auto master = getMaster();
@@ -134,9 +134,11 @@ class Monster final : public Creature {
this->spawnMonster = newSpawnMonster;
}
- int32_t getReflectPercent(CombatType_t combatType, bool useCharges = false) const override;
+ double_t getReflectPercent(CombatType_t combatType, bool useCharges = false) const override;
uint32_t getHealingCombatValue(CombatType_t healingType) const;
+ void addReflectElement(CombatType_t combatType, int32_t percent);
+
bool canWalkOnFieldType(CombatType_t combatType) const;
void onAttackedCreatureDisappear(bool isLogout) override;
@@ -144,6 +146,8 @@ class Monster final : public Creature {
void onRemoveCreature(std::shared_ptr creature, bool isLogout) override;
void onCreatureMove(const std::shared_ptr &creature, const std::shared_ptr &newTile, const Position &newPos, const std::shared_ptr &oldTile, const Position &oldPos, bool teleport) override;
void onCreatureSay(std::shared_ptr creature, SpeakClasses type, const std::string &text) override;
+ void onAttackedByPlayer(std::shared_ptr attackerPlayer);
+ void onSpawn();
void drainHealth(std::shared_ptr attacker, int32_t damage) override;
void changeHealth(int32_t healthChange, bool sendHealthChange = true) override;
@@ -333,6 +337,12 @@ class Monster final : public Creature {
bool isImmune(ConditionType_t conditionType) const override;
bool isImmune(CombatType_t combatType) const override;
+ void setImmune(bool immune) {
+ m_isImmune = immune;
+ }
+ bool isImmune() const {
+ return m_isImmune;
+ }
float getAttackMultiplier() const {
float multiplier = mType->getAttackMultiplier();
@@ -347,6 +357,14 @@ class Monster final : public Creature {
return multiplier * std::pow(1.02f, getForgeStack());
}
+ bool isDead() const override {
+ return m_isDead;
+ }
+
+ void setDead(bool isDead) {
+ m_isDead = isDead;
+ }
+
private:
auto getTargetIterator(const std::shared_ptr &creature) {
return std::ranges::find_if(targetList.begin(), targetList.end(), [id = creature->getID()](const std::weak_ptr &ref) {
@@ -372,6 +390,8 @@ class Monster final : public Creature {
int64_t lastMeleeAttack = 0;
+ uint16_t totalPlayersOnScreen = 0;
+
uint32_t attackTicks = 0;
uint32_t targetChangeTicks = 0;
uint32_t defenseTicks = 0;
@@ -385,8 +405,10 @@ class Monster final : public Creature {
int32_t stepDuration = 0;
int32_t targetDistance = 1;
int32_t challengeMeleeDuration = 0;
- uint16_t totalPlayersOnScreen = 0;
int32_t runAwayHealth = 0;
+ int32_t m_defense = 0;
+
+ std::unordered_map m_reflectElementMap;
Position masterPos;
@@ -402,6 +424,9 @@ class Monster final : public Creature {
bool hazardDamageBoost = false;
bool hazardDefenseBoost = false;
+ bool m_isDead = false;
+ bool m_isImmune = false;
+
void onCreatureEnter(std::shared_ptr creature);
void onCreatureLeave(std::shared_ptr creature);
void onCreatureFound(std::shared_ptr creature, bool pushFront = false);
diff --git a/src/creatures/monsters/monsters.cpp b/src/creatures/monsters/monsters.cpp
index 78358f69d87..7e192a79575 100644
--- a/src/creatures/monsters/monsters.cpp
+++ b/src/creatures/monsters/monsters.cpp
@@ -278,17 +278,34 @@ bool MonsterType::loadCallback(LuaScriptInterface* scriptInterface) {
}
info.scriptInterface = scriptInterface;
- if (info.eventType == MONSTERS_EVENT_THINK) {
- info.thinkEvent = id;
- } else if (info.eventType == MONSTERS_EVENT_APPEAR) {
- info.creatureAppearEvent = id;
- } else if (info.eventType == MONSTERS_EVENT_DISAPPEAR) {
- info.creatureDisappearEvent = id;
- } else if (info.eventType == MONSTERS_EVENT_MOVE) {
- info.creatureMoveEvent = id;
- } else if (info.eventType == MONSTERS_EVENT_SAY) {
- info.creatureSayEvent = id;
+
+ switch (info.eventType) {
+ case MONSTERS_EVENT_THINK:
+ info.thinkEvent = id;
+ break;
+ case MONSTERS_EVENT_APPEAR:
+ info.creatureAppearEvent = id;
+ break;
+ case MONSTERS_EVENT_DISAPPEAR:
+ info.creatureDisappearEvent = id;
+ break;
+ case MONSTERS_EVENT_MOVE:
+ info.creatureMoveEvent = id;
+ break;
+ case MONSTERS_EVENT_SAY:
+ info.creatureSayEvent = id;
+ break;
+ case MONSTERS_EVENT_ATTACKED_BY_PLAYER:
+ info.monsterAttackedByPlayerEvent = id;
+ break;
+ case MONSTERS_EVENT_ON_SPAWN:
+ info.spawnEvent = id;
+ break;
+ default:
+ g_logger().error("[MonsterType::loadCallback] - Unknown event type");
+ return false;
}
+
return true;
}
diff --git a/src/creatures/monsters/monsters.hpp b/src/creatures/monsters/monsters.hpp
index 171bef68e38..88d25e195c6 100644
--- a/src/creatures/monsters/monsters.hpp
+++ b/src/creatures/monsters/monsters.hpp
@@ -120,7 +120,9 @@ class MonsterType {
int32_t creatureDisappearEvent = -1;
int32_t creatureMoveEvent = -1;
int32_t creatureSayEvent = -1;
+ int32_t monsterAttackedByPlayerEvent = -1;
int32_t thinkEvent = -1;
+ int32_t spawnEvent = -1;
int32_t targetDistance = 1;
int32_t runAwayHealth = 0;
int32_t health = 100;
diff --git a/src/creatures/monsters/spawns/spawn_monster.cpp b/src/creatures/monsters/spawns/spawn_monster.cpp
index ca8f4584775..846068e67b7 100644
--- a/src/creatures/monsters/spawns/spawn_monster.cpp
+++ b/src/creatures/monsters/spawns/spawn_monster.cpp
@@ -177,7 +177,7 @@ bool SpawnMonster::spawnMonster(uint32_t spawnMonsterId, spawnBlock_t &sb, const
return false;
}
} else {
- g_logger().debug("[SpawnMonster] Spawning {} at {}", monsterType->name, sb.pos.toString());
+ g_logger().trace("[SpawnMonster] Spawning {} at {}", monsterType->name, sb.pos.toString());
if (!g_game().placeCreature(monster, sb.pos, false, true)) {
return false;
}
@@ -190,6 +190,7 @@ bool SpawnMonster::spawnMonster(uint32_t spawnMonsterId, spawnBlock_t &sb, const
spawnedMonsterMap[spawnMonsterId] = monster;
sb.lastSpawn = OTSYS_TIME();
g_events().eventMonsterOnSpawn(monster, sb.pos);
+ monster->onSpawn();
g_callbacks().executeCallback(EventCallback_t::monsterOnSpawn, &EventCallback::monsterOnSpawn, monster, sb.pos);
return true;
}
diff --git a/src/creatures/npcs/npc.cpp b/src/creatures/npcs/npc.cpp
index 18ca0dd0758..6d38524aef3 100644
--- a/src/creatures/npcs/npc.cpp
+++ b/src/creatures/npcs/npc.cpp
@@ -234,8 +234,8 @@ void Npc::onPlayerBuyItem(std::shared_ptr player, uint16_t itemId, uint8
return;
}
- // Check if the player not have empty slots
- if (!ignore && player->getFreeBackpackSlots() == 0) {
+ // Check if the player not have empty slots or the item is not a container
+ if (!ignore && (player->getFreeBackpackSlots() == 0 && (player->getInventoryItem(CONST_SLOT_BACKPACK) || (!Item::items[itemId].isContainer() || !(Item::items[itemId].slotPosition & SLOTP_BACKPACK))))) {
player->sendCancelMessage(RETURNVALUE_NOTENOUGHROOM);
return;
}
diff --git a/src/creatures/players/grouping/party.cpp b/src/creatures/players/grouping/party.cpp
index 76f1d955f10..b56e0c68605 100644
--- a/src/creatures/players/grouping/party.cpp
+++ b/src/creatures/players/grouping/party.cpp
@@ -454,7 +454,7 @@ void Party::shareExperience(uint64_t experience, std::shared_ptr targe
uint64_t shareExperience = experience;
g_events().eventPartyOnShareExperience(getParty(), shareExperience);
- g_callbacks().executeCallback(EventCallback_t::partyOnShareExperience, &EventCallback::partyOnShareExperience, getParty(), shareExperience);
+ g_callbacks().executeCallback(EventCallback_t::partyOnShareExperience, &EventCallback::partyOnShareExperience, getParty(), std::ref(shareExperience));
for (const auto &member : getMembers()) {
member->onGainSharedExperience(shareExperience, target);
diff --git a/src/creatures/players/player.cpp b/src/creatures/players/player.cpp
index 23b3fff7b95..c4ef6de6787 100644
--- a/src/creatures/players/player.cpp
+++ b/src/creatures/players/player.cpp
@@ -677,7 +677,7 @@ void Player::addSkillAdvance(skills_t skill, uint64_t count) {
}
g_events().eventPlayerOnGainSkillTries(static_self_cast(), skill, count);
- g_callbacks().executeCallback(EventCallback_t::playerOnGainSkillTries, &EventCallback::playerOnGainSkillTries, getPlayer(), skill, count);
+ g_callbacks().executeCallback(EventCallback_t::playerOnGainSkillTries, &EventCallback::playerOnGainSkillTries, getPlayer(), std::ref(skill), std::ref(count));
if (count == 0) {
return;
}
@@ -2228,6 +2228,8 @@ void Player::onThink(uint32_t interval) {
// Wheel of destiny major spells
wheel()->onThink();
+
+ g_callbacks().executeCallback(EventCallback_t::playerOnThink, &EventCallback::playerOnThink, getPlayer(), interval);
}
uint32_t Player::isMuted() const {
@@ -2361,7 +2363,7 @@ void Player::addExperience(std::shared_ptr target, uint64_t exp, bool
return;
}
- g_callbacks().executeCallback(EventCallback_t::playerOnGainExperience, &EventCallback::playerOnGainExperience, getPlayer(), target, exp, rawExp);
+ g_callbacks().executeCallback(EventCallback_t::playerOnGainExperience, &EventCallback::playerOnGainExperience, getPlayer(), target, std::ref(exp), std::ref(rawExp));
g_events().eventPlayerOnGainExperience(static_self_cast(), target, exp, rawExp);
if (exp == 0) {
@@ -2477,7 +2479,7 @@ void Player::removeExperience(uint64_t exp, bool sendText /* = false*/) {
}
g_events().eventPlayerOnLoseExperience(static_self_cast(), exp);
- g_callbacks().executeCallback(EventCallback_t::playerOnLoseExperience, &EventCallback::playerOnLoseExperience, getPlayer(), exp);
+ g_callbacks().executeCallback(EventCallback_t::playerOnLoseExperience, &EventCallback::playerOnLoseExperience, getPlayer(), std::ref(exp));
if (exp == 0) {
return;
}
@@ -5553,15 +5555,15 @@ int32_t Player::getMagicShieldCapacityPercent(bool useCharges) const {
return result;
}
-int32_t Player::getReflectPercent(CombatType_t combat, bool useCharges) const {
- int32_t result = reflectPercent[combatTypeToIndex(combat)];
- for (const auto &item : getEquippedItems()) {
+double_t Player::getReflectPercent(CombatType_t combat, bool useCharges) const {
+ double_t result = reflectPercent[combatTypeToIndex(combat)];
+ for (const auto item : getEquippedItems()) {
const ItemType &itemType = Item::items[item->getID()];
if (!itemType.abilities) {
continue;
}
- int32_t reflectPercent = itemType.abilities->reflectPercent[combatTypeToIndex(combat)];
+ double_t reflectPercent = itemType.abilities->reflectPercent[combatTypeToIndex(combat)];
if (reflectPercent != 0) {
result += reflectPercent;
uint16_t charges = item->getCharges();
@@ -5959,7 +5961,7 @@ bool Player::addOfflineTrainingTries(skills_t skill, uint64_t tries) {
oldPercentToNextLevel = static_cast(manaSpent * 100) / nextReqMana;
g_events().eventPlayerOnGainSkillTries(static_self_cast(), SKILL_MAGLEVEL, tries);
- g_callbacks().executeCallback(EventCallback_t::playerOnGainSkillTries, &EventCallback::playerOnGainSkillTries, getPlayer(), SKILL_MAGLEVEL, tries);
+ g_callbacks().executeCallback(EventCallback_t::playerOnGainSkillTries, &EventCallback::playerOnGainSkillTries, getPlayer(), SKILL_MAGLEVEL, std::ref(tries));
uint32_t currMagLevel = magLevel;
while ((manaSpent + tries) >= nextReqMana) {
@@ -6177,6 +6179,20 @@ void Player::sendIconBakragore(const IconBakragore icon) {
}
}
+void Player::removeBakragoreIcons() {
+ for (auto icon : magic_enum::enum_values()) {
+ if (hasCondition(CONDITION_BAKRAGORE, enumToValue(icon))) {
+ removeCondition(CONDITION_BAKRAGORE, CONDITIONID_DEFAULT, true);
+ }
+ }
+}
+
+void Player::removeBakragoreIcon(const IconBakragore icon) {
+ if (hasCondition(CONDITION_BAKRAGORE, enumToValue(icon))) {
+ removeCondition(CONDITION_BAKRAGORE, CONDITIONID_DEFAULT, true);
+ }
+}
+
void Player::sendCyclopediaCharacterAchievements(uint16_t secretsUnlocked, std::vector> achievementsUnlocked) {
if (client) {
client->sendCyclopediaCharacterAchievements(secretsUnlocked, achievementsUnlocked);
diff --git a/src/creatures/players/player.hpp b/src/creatures/players/player.hpp
index e1242c38a1b..b2e49afa1d0 100644
--- a/src/creatures/players/player.hpp
+++ b/src/creatures/players/player.hpp
@@ -417,7 +417,7 @@ class Player final : public Creature, public Cylinder, public Bankable {
magicShieldCapacityPercent += value;
}
- int32_t getReflectPercent(CombatType_t combat, bool useCharges = false) const override;
+ double_t getReflectPercent(CombatType_t combat, bool useCharges = false) const override;
int32_t getReflectFlat(CombatType_t combat, bool useCharges = false) const override;
@@ -1422,6 +1422,8 @@ class Player final : public Creature, public Cylinder, public Bankable {
void sendClosePrivate(uint16_t channelId);
void sendIcons();
void sendIconBakragore(const IconBakragore icon);
+ void removeBakragoreIcons();
+ void removeBakragoreIcon(const IconBakragore icon);
void sendClientCheck() const {
if (client) {
client->sendClientCheck();
@@ -2943,7 +2945,7 @@ class Player final : public Creature, public Cylinder, public Bankable {
bool marketMenu = false; // Menu option 'show in market'
bool exerciseTraining = false;
bool moved = false;
- bool dead = false;
+ bool m_isDead = false;
bool imbuementTrackerWindowOpen = false;
// Hazard system
@@ -3013,10 +3015,10 @@ class Player final : public Creature, public Cylinder, public Bankable {
void getPathSearchParams(const std::shared_ptr &creature, FindPathParams &fpp) override;
void setDead(bool isDead) {
- dead = isDead;
+ m_isDead = isDead;
}
- bool isDead() const {
- return dead;
+ bool isDead() const override {
+ return m_isDead;
}
void triggerMomentum();
diff --git a/src/enums/player_icons.hpp b/src/enums/player_icons.hpp
index c289144bd7b..7878d9e5037 100644
--- a/src/enums/player_icons.hpp
+++ b/src/enums/player_icons.hpp
@@ -35,11 +35,11 @@ enum class PlayerIcon : uint8_t {
GreaterHex = 18,
Rooted = 19,
Feared = 20,
- Goshnar1 = 21,
- Goshnar2 = 22,
- Goshnar3 = 23,
- Goshnar4 = 24,
- Goshnar5 = 25,
+ GoshnarTaint1 = 21,
+ GoshnarTaint2 = 22,
+ GoshnarTaint3 = 23,
+ GoshnarTaint4 = 24,
+ GoshnarTaint5 = 25,
NewManaShield = 26,
Agony = 27,
diff --git a/src/game/bank/bank.cpp b/src/game/bank/bank.cpp
index d9a056396ce..b6f3b14c6e5 100644
--- a/src/game/bank/bank.cpp
+++ b/src/game/bank/bank.cpp
@@ -130,6 +130,10 @@ bool Bank::transferTo(const std::shared_ptr destination, uint64_t amount)
}
bool Bank::withdraw(std::shared_ptr player, uint64_t amount) {
+ if (!player) {
+ return false;
+ }
+
if (!debit(amount)) {
return false;
}
diff --git a/src/game/game.cpp b/src/game/game.cpp
index df1b401bf48..7b60220ecf9 100644
--- a/src/game/game.cpp
+++ b/src/game/game.cpp
@@ -1165,6 +1165,7 @@ bool Game::removeCreature(std::shared_ptr creature, bool isLogout /* =
size_t i = 0;
for (const auto &spectator : playersSpectators) {
if (const auto &player = spectator->getPlayer()) {
+ player->sendMagicEffect(tilePosition, CONST_ME_POFF);
player->sendRemoveTileThing(tilePosition, oldStackPosVector[i++]);
}
}
@@ -1476,6 +1477,10 @@ ReturnValue Game::internalMoveCreature(std::shared_ptr creature, Direc
return RETURNVALUE_NOTPOSSIBLE;
}
+ if (creature->getBaseSpeed() == 0) {
+ return RETURNVALUE_NOTMOVABLE;
+ }
+
creature->setLastPosition(creature->getPosition());
const Position ¤tPos = creature->getPosition();
Position destPos = getNextPosition(direction, currentPos);
@@ -7109,7 +7114,7 @@ bool Game::combatChangeHealth(std::shared_ptr attacker, std::shared_pt
if (!isEvent) {
g_events().eventCreatureOnDrainHealth(target, attacker, damage.primary.type, damage.primary.value, damage.secondary.type, damage.secondary.value, message.primary.color, message.secondary.color);
- g_callbacks().executeCallback(EventCallback_t::creatureOnDrainHealth, &EventCallback::creatureOnDrainHealth, target, attacker, damage.primary.type, damage.primary.value, damage.secondary.type, damage.secondary.value, message.primary.color, message.secondary.color);
+ g_callbacks().executeCallback(EventCallback_t::creatureOnDrainHealth, &EventCallback::creatureOnDrainHealth, target, attacker, std::ref(damage.primary.type), std::ref(damage.primary.value), std::ref(damage.secondary.type), std::ref(damage.secondary.value), std::ref(message.primary.color), std::ref(message.secondary.color));
}
if (damage.origin != ORIGIN_NONE && attacker && damage.primary.type != COMBAT_HEALING) {
damage.primary.value *= attacker->getBuff(BUFF_DAMAGEDEALT) / 100.;
@@ -7297,7 +7302,7 @@ bool Game::combatChangeHealth(std::shared_ptr attacker, std::shared_pt
return true;
} else if (realDamage >= targetHealth) {
for (const auto &creatureEvent : target->getCreatureEvents(CREATURE_EVENT_PREPAREDEATH)) {
- if (!creatureEvent->executeOnPrepareDeath(target, attacker)) {
+ if (!creatureEvent->executeOnPrepareDeath(target, attacker, std::ref(realDamage))) {
return false;
}
}
diff --git a/src/game/zones/zone.cpp b/src/game/zones/zone.cpp
index b6c1191aa78..7ffa4b594f4 100644
--- a/src/game/zones/zone.cpp
+++ b/src/game/zones/zone.cpp
@@ -15,6 +15,7 @@
#include "creatures/npcs/npc.hpp"
#include "creatures/players/player.hpp"
#include "utils/pugicast.hpp"
+#include "kv/kv.hpp"
phmap::parallel_flat_hash_map> Zone::zones = {};
phmap::parallel_flat_hash_map> Zone::zonesByID = {};
@@ -122,6 +123,10 @@ std::vector> Zone::getItems() {
void Zone::removePlayers() {
for (const auto &player : getPlayers()) {
g_game().internalTeleport(player, getRemoveDestination(player));
+ // Remove icon from player (soul war quest)
+ if (player->hasIcon("goshnars-hatred-damage")) {
+ player->removeIcon("goshnars-hatred-damage");
+ }
}
}
diff --git a/src/io/functions/iologindata_load_player.cpp b/src/io/functions/iologindata_load_player.cpp
index 6dfdef273da..df6a202a59b 100644
--- a/src/io/functions/iologindata_load_player.cpp
+++ b/src/io/functions/iologindata_load_player.cpp
@@ -846,7 +846,12 @@ void IOLoginDataLoad::loadPlayerForgeHistory(std::shared_ptr player, DBR
}
void IOLoginDataLoad::loadPlayerBosstiary(std::shared_ptr player, DBResult_ptr result) {
- if (!result || !player) {
+ if (!result) {
+ g_logger().warn("[IOLoginData::loadPlayer] - Result nullptr: {}", __FUNCTION__);
+ return;
+ }
+
+ if (!player) {
g_logger().warn("[IOLoginData::loadPlayer] - Player or Result nullptr: {}", __FUNCTION__);
return;
}
diff --git a/src/io/io_bosstiary.cpp b/src/io/io_bosstiary.cpp
index 2f625cdb7cc..7c7c8108ce7 100644
--- a/src/io/io_bosstiary.cpp
+++ b/src/io/io_bosstiary.cpp
@@ -249,6 +249,7 @@ std::vector IOBosstiary::getBosstiaryFinished(const std::shared_ptrflush();
- g_logger().info("Map Loaded {} ({}x{}) in {} milliseconds", map->path.filename().string(), map->width, map->height, bm_mapLoad.duration());
+ g_logger().debug("Map Loaded {} ({}x{}) in {} milliseconds", map->path.filename().string(), map->width, map->height, bm_mapLoad.duration());
}
void IOMap::parseMapDataAttributes(FileStream &stream, Map* map) {
diff --git a/src/lua/callbacks/callbacks_definitions.hpp b/src/lua/callbacks/callbacks_definitions.hpp
index 3b8016f5f5b..6c19cc809c9 100644
--- a/src/lua/callbacks/callbacks_definitions.hpp
+++ b/src/lua/callbacks/callbacks_definitions.hpp
@@ -25,6 +25,7 @@ enum class EventCallback_t : uint16_t {
creatureOnTargetCombat,
creatureOnHear,
creatureOnDrainHealth,
+ creatureOnCombat,
// Party
partyOnJoin,
partyOnLeave,
@@ -57,6 +58,7 @@ enum class EventCallback_t : uint16_t {
playerOnInventoryUpdate,
playerOnRotateItem,
playerOnWalk,
+ playerOnThink,
// Monster
monsterOnDropLoot,
monsterPostDropLoot,
@@ -68,4 +70,5 @@ enum class EventCallback_t : uint16_t {
zoneBeforeCreatureLeave,
zoneAfterCreatureEnter,
zoneAfterCreatureLeave,
+ mapOnLoad,
};
diff --git a/src/lua/callbacks/event_callback.cpp b/src/lua/callbacks/event_callback.cpp
index 38e7654d8d5..8280bad6d2b 100644
--- a/src/lua/callbacks/event_callback.cpp
+++ b/src/lua/callbacks/event_callback.cpp
@@ -25,8 +25,16 @@
*
* @see Script
*/
-EventCallback::EventCallback(LuaScriptInterface* scriptInterface) :
- Script(scriptInterface) {
+EventCallback::EventCallback(LuaScriptInterface* scriptInterface, const std::string &callbackName, bool skipDuplicationCheck) :
+ Script(scriptInterface), m_callbackName(callbackName), m_skipDuplicationCheck(skipDuplicationCheck) {
+}
+
+std::string EventCallback::getName() const {
+ return m_callbackName;
+}
+
+bool EventCallback::skipDuplicationCheck() const {
+ return m_skipDuplicationCheck;
}
std::string EventCallback::getScriptTypeName() const {
@@ -225,6 +233,58 @@ void EventCallback::creatureOnDrainHealth(std::shared_ptr creature, st
getScriptInterface()->resetScriptEnv();
}
+void EventCallback::creatureOnCombat(std::shared_ptr attacker, std::shared_ptr target, CombatDamage &damage) const {
+ if (!getScriptInterface()->reserveScriptEnv()) {
+ g_logger().error("[{} - "
+ "Creature {} target {}] "
+ "Call stack overflow. Too many lua script calls being nested.",
+ __FUNCTION__, attacker->getName(), target->getName());
+ return;
+ }
+
+ ScriptEnvironment* scriptEnvironment = getScriptInterface()->getScriptEnv();
+ scriptEnvironment->setScriptId(getScriptId(), getScriptInterface());
+
+ lua_State* L = getScriptInterface()->getLuaState();
+ getScriptInterface()->pushFunction(getScriptId());
+
+ LuaScriptInterface::pushUserdata(L, attacker);
+ LuaScriptInterface::setCreatureMetatable(L, -1, attacker);
+
+ LuaScriptInterface::pushUserdata(L, target);
+ LuaScriptInterface::setCreatureMetatable(L, -1, target);
+
+ LuaScriptInterface::pushCombatDamage(L, damage);
+
+ if (getScriptInterface()->protectedCall(L, 7, 4) != 0) {
+ LuaScriptInterface::reportError(nullptr, LuaScriptInterface::popString(L));
+ } else {
+ damage.primary.value = std::abs(LuaScriptInterface::getNumber(L, -4));
+ damage.primary.type = LuaScriptInterface::getNumber(L, -3);
+ damage.secondary.value = std::abs(LuaScriptInterface::getNumber(L, -2));
+ damage.secondary.type = LuaScriptInterface::getNumber(L, -1);
+
+ lua_pop(L, 4);
+ if (damage.primary.type != COMBAT_HEALING) {
+ damage.primary.value = -damage.primary.value;
+ damage.secondary.value = -damage.secondary.value;
+ }
+ /*
+ Only EK with dealing physical damage will get elemental damage on skill
+ */
+ if (damage.origin == ORIGIN_SPELL && attacker) {
+ const auto &player = attacker->getPlayer();
+ if (player && player->getVocationId() != 4 && player->getVocationId() != 8) {
+ damage.primary.value = damage.primary.value + damage.secondary.value;
+ damage.secondary.type = COMBAT_NONE;
+ damage.secondary.value = 0;
+ }
+ }
+ }
+
+ getScriptInterface()->resetScriptEnv();
+}
+
// Party
bool EventCallback::partyOnJoin(std::shared_ptr party, std::shared_ptr player) const {
if (!getScriptInterface()->reserveScriptEnv()) {
@@ -852,7 +912,7 @@ void EventCallback::playerOnCombat(std::shared_ptr player, std::shared_p
if (target) {
LuaScriptInterface::pushUserdata(L, target);
- LuaScriptInterface::setMetatable(L, -1, "Creature");
+ LuaScriptInterface::setCreatureMetatable(L, -1, target);
} else {
lua_pushnil(L);
}
@@ -1034,6 +1094,26 @@ void EventCallback::playerOnStorageUpdate(std::shared_ptr player, const
getScriptInterface()->callVoidFunction(5);
}
+void EventCallback::playerOnThink(std::shared_ptr player, uint32_t interval) const {
+ if (!getScriptInterface()->reserveScriptEnv()) {
+ g_logger().error("[{}] player {}. Call stack overflow. Too many lua script calls being nested.", __FUNCTION__, player->getName());
+ return;
+ }
+
+ ScriptEnvironment* scriptEnvironment = getScriptInterface()->getScriptEnv();
+ scriptEnvironment->setScriptId(getScriptId(), getScriptInterface());
+
+ lua_State* L = getScriptInterface()->getLuaState();
+ getScriptInterface()->pushFunction(getScriptId());
+
+ LuaScriptInterface::pushUserdata(L, player);
+ LuaScriptInterface::setMetatable(L, -1, "Player");
+
+ lua_pushnumber(L, interval);
+
+ getScriptInterface()->callVoidFunction(2);
+}
+
// Monster
void EventCallback::monsterOnDropLoot(std::shared_ptr monster, std::shared_ptr corpse) const {
if (!getScriptInterface()->reserveScriptEnv()) {
@@ -1235,3 +1315,22 @@ void EventCallback::zoneAfterCreatureLeave(std::shared_ptr zone, std::shar
getScriptInterface()->callVoidFunction(2);
}
+
+void EventCallback::mapOnLoad(const std::string &mapFullPath) const {
+ if (!getScriptInterface()->reserveScriptEnv()) {
+ g_logger().error("[{} - "
+ "Call stack overflow. Too many lua script calls being nested.",
+ __FUNCTION__);
+ return;
+ }
+
+ ScriptEnvironment* scriptEnvironment = getScriptInterface()->getScriptEnv();
+ scriptEnvironment->setScriptId(getScriptId(), getScriptInterface());
+
+ lua_State* L = getScriptInterface()->getLuaState();
+ getScriptInterface()->pushFunction(getScriptId());
+
+ LuaScriptInterface::pushString(L, mapFullPath);
+
+ getScriptInterface()->callVoidFunction(1);
+}
diff --git a/src/lua/callbacks/event_callback.hpp b/src/lua/callbacks/event_callback.hpp
index 9141235a028..9e65480c246 100644
--- a/src/lua/callbacks/event_callback.hpp
+++ b/src/lua/callbacks/event_callback.hpp
@@ -35,13 +35,27 @@ class EventCallback : public Script {
private:
EventCallback_t m_callbackType = EventCallback_t::none; ///< The type of the event callback.
std::string m_scriptTypeName; ///< The name associated with the script type.
+ std::string m_callbackName; ///< The name of the callback.
+ bool m_skipDuplicationCheck = false; ///< Whether the callback is silent error for already registered log error.
public:
/**
* @brief Constructor that initializes the EventCallback with a given script interface.
* @param scriptInterface Pointer to the LuaScriptInterface object.
*/
- explicit EventCallback(LuaScriptInterface* scriptInterface);
+ explicit EventCallback(LuaScriptInterface* scriptInterface, const std::string &callbackName, bool silentAlreadyRegistered);
+
+ /**
+ * @brief Retrieves the callback name.
+ * @return The callback name as a string.
+ */
+ std::string getName() const;
+
+ /**
+ * @brief Retrieves the skip registration status of the callback.
+ * @return True if the callback is true for skip duplication check and register again the event, false otherwise.
+ */
+ bool skipDuplicationCheck() const;
/**
* @brief Retrieves the script type name.
@@ -84,6 +98,7 @@ class EventCallback : public Script {
ReturnValue creatureOnTargetCombat(std::shared_ptr creature, std::shared_ptr target) const;
void creatureOnHear(std::shared_ptr creature, std::shared_ptr speaker, const std::string &words, SpeakClasses type) const;
void creatureOnDrainHealth(std::shared_ptr creature, std::shared_ptr attacker, CombatType_t &typePrimary, int32_t &damagePrimary, CombatType_t &typeSecondary, int32_t &damageSecondary, TextColor_t &colorPrimary, TextColor_t &colorSecondary) const;
+ void creatureOnCombat(std::shared_ptr attacker, std::shared_ptr target, CombatDamage &damage) const;
// Party
bool partyOnJoin(std::shared_ptr party, std::shared_ptr player) const;
@@ -117,6 +132,7 @@ class EventCallback : public Script {
void playerOnInventoryUpdate(std::shared_ptr player, std::shared_ptr
- item, Slots_t slot, bool equip) const;
bool playerOnRotateItem(std::shared_ptr player, std::shared_ptr
- item, const Position &position) const;
void playerOnWalk(std::shared_ptr player, Direction &dir) const;
+ void playerOnThink(std::shared_ptr player, uint32_t interval) const;
// Monster
void monsterOnDropLoot(std::shared_ptr monster, std::shared_ptr corpse) const;
@@ -132,7 +148,5 @@ class EventCallback : public Script {
void zoneAfterCreatureEnter(std::shared_ptr zone, std::shared_ptr creature) const;
void zoneAfterCreatureLeave(std::shared_ptr zone, std::shared_ptr creature) const;
- /**
- * @note here end the lua binder functions }
- */
+ void mapOnLoad(const std::string &mapFullPath) const;
};
diff --git a/src/lua/callbacks/events_callbacks.cpp b/src/lua/callbacks/events_callbacks.cpp
index 4a1830f80fb..13a42baa15a 100644
--- a/src/lua/callbacks/events_callbacks.cpp
+++ b/src/lua/callbacks/events_callbacks.cpp
@@ -12,6 +12,8 @@
#include "lua/callbacks/events_callbacks.hpp"
#include "lua/callbacks/event_callback.hpp"
+#include "game/game.hpp"
+#include "lib/di/container.hpp"
/**
* @class EventsCallbacks
@@ -28,22 +30,37 @@ EventsCallbacks &EventsCallbacks::getInstance() {
return inject();
}
-void EventsCallbacks::addCallback(const std::shared_ptr callback) {
- m_callbacks.push_back(callback);
+bool EventsCallbacks::isCallbackRegistered(const std::shared_ptr &callback) {
+ if (g_game().getGameState() == GAME_STATE_STARTUP && !callback->skipDuplicationCheck() && m_callbacks.find(callback->getName()) != m_callbacks.end()) {
+ return true;
+ }
+
+ return false;
+}
+
+void EventsCallbacks::addCallback(const std::shared_ptr &callback) {
+ if (m_callbacks.find(callback->getName()) != m_callbacks.end() && !callback->skipDuplicationCheck()) {
+ g_logger().trace("Event callback already registered: {}", callback->getName());
+ return;
+ }
+
+ g_logger().trace("Registering event callback: {}", callback->getName());
+
+ m_callbacks[callback->getName()] = callback;
}
-std::vector> EventsCallbacks::getCallbacks() const {
+std::unordered_map> EventsCallbacks::getCallbacks() const {
return m_callbacks;
}
-std::vector> EventsCallbacks::getCallbacksByType(EventCallback_t type) const {
- std::vector> eventCallbacks;
- for (auto callback : getCallbacks()) {
+std::unordered_map> EventsCallbacks::getCallbacksByType(EventCallback_t type) const {
+ std::unordered_map> eventCallbacks;
+ for (auto [name, callback] : getCallbacks()) {
if (callback->getType() != type) {
continue;
}
- eventCallbacks.push_back(callback);
+ eventCallbacks[name] = callback;
}
return eventCallbacks;
diff --git a/src/lua/callbacks/events_callbacks.hpp b/src/lua/callbacks/events_callbacks.hpp
index f71103047e6..dff5cec30c5 100644
--- a/src/lua/callbacks/events_callbacks.hpp
+++ b/src/lua/callbacks/events_callbacks.hpp
@@ -45,24 +45,35 @@ class EventsCallbacks {
*/
static EventsCallbacks &getInstance();
+ /**
+ * @brief Checks if an event callback is already registered.
+ *
+ * @details Determines if the game state is at startup and if a callback with the same name already exists.
+ * @details If both conditions are met, logs an error and indicates the callback is already registered.
+ *
+ * @param callback Shared pointer to the event callback being checked.
+ * @return True if the callback already exists during the game startup state, otherwise false.
+ */
+ bool isCallbackRegistered(const std::shared_ptr &callback);
+
/**
* @brief Adds a new event callback to the list.
* @param callback Pointer to the EventCallback object to add.
*/
- void addCallback(const std::shared_ptr callback);
+ void addCallback(const std::shared_ptr &callback);
/**
* @brief Gets all registered event callbacks.
* @return Vector of pointers to EventCallback objects.
*/
- std::vector> getCallbacks() const;
+ std::unordered_map> getCallbacks() const;
/**
* @brief Gets event callbacks by their type.
* @param type The type of callbacks to retrieve.
* @return Vector of pointers to EventCallback objects of the specified type.
*/
- std::vector> getCallbacksByType(EventCallback_t type) const;
+ std::unordered_map> getCallbacksByType(EventCallback_t type) const;
/**
* @brief Clears all registered event callbacks.
@@ -77,7 +88,7 @@ class EventsCallbacks {
*/
template
void executeCallback(EventCallback_t eventType, CallbackFunc callbackFunc, Args &&... args) {
- for (const auto &callback : getCallbacksByType(eventType)) {
+ for (const auto &[name, callback] : getCallbacksByType(eventType)) {
auto argsCopy = std::make_tuple(args...);
if (callback && callback->isLoadedCallback()) {
std::apply(
@@ -86,6 +97,7 @@ class EventsCallbacks {
},
argsCopy
);
+ g_logger().trace("Executed callback: {}", name);
}
}
}
@@ -99,7 +111,7 @@ class EventsCallbacks {
template
ReturnValue checkCallbackWithReturnValue(EventCallback_t eventType, CallbackFunc callbackFunc, Args &&... args) {
ReturnValue res = RETURNVALUE_NOERROR;
- for (const auto &callback : getCallbacksByType(eventType)) {
+ for (const auto &[name, callback] : getCallbacksByType(eventType)) {
auto argsCopy = std::make_tuple(args...);
if (callback && callback->isLoadedCallback()) {
ReturnValue callbackResult = std::apply(
@@ -127,7 +139,7 @@ class EventsCallbacks {
bool checkCallback(EventCallback_t eventType, CallbackFunc callbackFunc, Args &&... args) {
bool allCallbacksSucceeded = true;
- for (const auto &callback : getCallbacksByType(eventType)) {
+ for (const auto &[name, callback] : getCallbacksByType(eventType)) {
auto argsCopy = std::make_tuple(args...);
if (callback && callback->isLoadedCallback()) {
bool callbackResult = std::apply(
@@ -144,7 +156,7 @@ class EventsCallbacks {
private:
// Container for storing registered event callbacks.
- std::vector> m_callbacks;
+ std::unordered_map> m_callbacks;
};
constexpr auto g_callbacks = EventsCallbacks::getInstance;
diff --git a/src/lua/creature/creatureevent.cpp b/src/lua/creature/creatureevent.cpp
index 09dbdf086ce..90d09d2b7f9 100644
--- a/src/lua/creature/creatureevent.cpp
+++ b/src/lua/creature/creatureevent.cpp
@@ -231,7 +231,7 @@ bool CreatureEvent::executeOnThink(std::shared_ptr creature, uint32_t
return getScriptInterface()->callFunction(2);
}
-bool CreatureEvent::executeOnPrepareDeath(std::shared_ptr creature, std::shared_ptr killer) const {
+bool CreatureEvent::executeOnPrepareDeath(std::shared_ptr creature, std::shared_ptr killer, int realDamage) const {
// onPrepareDeath(creature, killer)
if (!getScriptInterface()->reserveScriptEnv()) {
g_logger().error("[CreatureEvent::executeOnPrepareDeath - Creature {} killer {}"
@@ -257,7 +257,9 @@ bool CreatureEvent::executeOnPrepareDeath(std::shared_ptr creature, st
lua_pushnil(L);
}
- return getScriptInterface()->callFunction(2);
+ lua_pushnumber(L, realDamage);
+
+ return getScriptInterface()->callFunction(3);
}
bool CreatureEvent::executeOnDeath(std::shared_ptr creature, std::shared_ptr
- corpse, std::shared_ptr killer, std::shared_ptr mostDamageKiller, bool lastHitUnjustified, bool mostDamageUnjustified) const {
diff --git a/src/lua/creature/creatureevent.hpp b/src/lua/creature/creatureevent.hpp
index 8a208343e11..e34bd8eb75e 100644
--- a/src/lua/creature/creatureevent.hpp
+++ b/src/lua/creature/creatureevent.hpp
@@ -46,7 +46,7 @@ class CreatureEvent final : public Script {
bool executeOnLogin(std::shared_ptr player) const;
bool executeOnLogout(std::shared_ptr player) const;
bool executeOnThink(std::shared_ptr creature, uint32_t interval) const;
- bool executeOnPrepareDeath(std::shared_ptr creature, std::shared_ptr killer) const;
+ bool executeOnPrepareDeath(std::shared_ptr creature, std::shared_ptr killer, int realDamage) const;
bool executeOnDeath(std::shared_ptr creature, std::shared_ptr
- corpse, std::shared_ptr killer, std::shared_ptr mostDamageKiller, bool lastHitUnjustified, bool mostDamageUnjustified) const;
void executeOnKill(std::shared_ptr creature, std::shared_ptr target, bool lastHit) const;
bool executeAdvance(std::shared_ptr player, skills_t, uint32_t, uint32_t) const;
diff --git a/src/lua/creature/events.cpp b/src/lua/creature/events.cpp
index 8bdd53465b3..144c188e900 100644
--- a/src/lua/creature/events.cpp
+++ b/src/lua/creature/events.cpp
@@ -1159,7 +1159,7 @@ void Events::eventPlayerOnCombat(std::shared_ptr player, std::shared_ptr
if (target) {
LuaScriptInterface::pushUserdata(L, target);
- LuaScriptInterface::setMetatable(L, -1, "Creature");
+ LuaScriptInterface::setCreatureMetatable(L, -1, target);
} else {
lua_pushnil(L);
}
diff --git a/src/lua/functions/core/game/bank_functions.cpp b/src/lua/functions/core/game/bank_functions.cpp
index f6732b8bf70..cc76bf4c091 100644
--- a/src/lua/functions/core/game/bank_functions.cpp
+++ b/src/lua/functions/core/game/bank_functions.cpp
@@ -81,6 +81,7 @@ int BankFunctions::luaBankTransferToGuild(lua_State* L) {
reportErrorFunc("Source is nullptr");
return 1;
}
+
std::shared_ptr destination = getBank(L, 2, true /* isGuild */);
if (destination == nullptr) {
reportErrorFunc("Destination is nullptr");
diff --git a/src/lua/functions/core/game/game_functions.cpp b/src/lua/functions/core/game/game_functions.cpp
index 83ecd091850..10a6379781a 100644
--- a/src/lua/functions/core/game/game_functions.cpp
+++ b/src/lua/functions/core/game/game_functions.cpp
@@ -449,6 +449,7 @@ int GameFunctions::luaGameCreateMonster(lua_State* L) {
if (g_game().placeCreature(monster, position, extended, force)) {
g_events().eventMonsterOnSpawn(monster, position);
g_callbacks().executeCallback(EventCallback_t::monsterOnSpawn, &EventCallback::monsterOnSpawn, monster, position);
+ monster->onSpawn();
const auto &mtype = monster->getMonsterType();
if (mtype && mtype->info.raceid > 0 && mtype->info.bosstiaryRace == BosstiaryRarity_t::RARITY_ARCHFOE) {
for (const auto &spectator : Spectators().find(monster->getPosition(), true)) {
diff --git a/src/lua/functions/core/game/lua_enums.cpp b/src/lua/functions/core/game/lua_enums.cpp
index 05fc72c019f..e4dc252be0d 100644
--- a/src/lua/functions/core/game/lua_enums.cpp
+++ b/src/lua/functions/core/game/lua_enums.cpp
@@ -1206,11 +1206,9 @@ void LuaEnums::initReloadTypeEnums(lua_State* L) {
void LuaEnums::initCreaturesEventEnums(lua_State* L) {
// Monsters
- registerEnum(L, MONSTERS_EVENT_THINK);
- registerEnum(L, MONSTERS_EVENT_APPEAR);
- registerEnum(L, MONSTERS_EVENT_DISAPPEAR);
- registerEnum(L, MONSTERS_EVENT_MOVE);
- registerEnum(L, MONSTERS_EVENT_SAY);
+ for (auto value : magic_enum::enum_values()) {
+ registerMagicEnum(L, value);
+ }
// Npcs
registerEnum(L, NPCS_EVENT_THINK);
diff --git a/src/lua/functions/core/game/zone_functions.cpp b/src/lua/functions/core/game/zone_functions.cpp
index 5471d01895a..0f19c94f789 100644
--- a/src/lua/functions/core/game/zone_functions.cpp
+++ b/src/lua/functions/core/game/zone_functions.cpp
@@ -136,7 +136,7 @@ int ZoneFunctions::luaZoneGetCreatures(lua_State* L) {
for (auto creature : creatures) {
index++;
pushUserdata(L, creature);
- setMetatable(L, -1, "Creature");
+ setCreatureMetatable(L, -1, creature);
lua_rawseti(L, -2, index);
}
return 1;
diff --git a/src/lua/functions/creatures/monster/monster_functions.cpp b/src/lua/functions/creatures/monster/monster_functions.cpp
index 5578477d6ba..917e90f85f4 100644
--- a/src/lua/functions/creatures/monster/monster_functions.cpp
+++ b/src/lua/functions/creatures/monster/monster_functions.cpp
@@ -60,7 +60,8 @@ int MonsterFunctions::luaMonsterGetType(lua_State* L) {
}
int MonsterFunctions::luaMonsterSetType(lua_State* L) {
- // monster:setType(name or raceid)
+ // monster:setType(name or raceid, restoreHealth = false)
+ bool restoreHealth = getBoolean(L, 3, false);
std::shared_ptr monster = getUserdataShared(L, 1);
if (monster) {
std::shared_ptr mType = nullptr;
@@ -81,8 +82,14 @@ int MonsterFunctions::luaMonsterSetType(lua_State* L) {
monster->defaultOutfit = mType->info.outfit;
monster->currentOutfit = mType->info.outfit;
monster->skull = mType->info.skull;
- monster->health = mType->info.health * mType->getHealthMultiplier();
- monster->healthMax = mType->info.healthMax * mType->getHealthMultiplier();
+ if (restoreHealth) {
+ auto multiplier = mType->getHealthMultiplier();
+ monster->health = mType->info.health * multiplier;
+ monster->healthMax = mType->info.healthMax * multiplier;
+ } else {
+ monster->health = monster->getHealth();
+ monster->healthMax = monster->getMaxHealth();
+ }
monster->baseSpeed = mType->getBaseSpeed();
monster->internalLight = mType->info.light;
monster->hiddenHealth = mType->info.hiddenHealth;
@@ -630,3 +637,76 @@ int MonsterFunctions::luaMonsterHazardDefenseBoost(lua_State* L) {
}
return 1;
}
+
+int MonsterFunctions::luaMonsterAddReflectElement(lua_State* L) {
+ // monster:addReflectElement(type, percent)
+ const auto &monster = getUserdataShared(L, 1);
+ if (!monster) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_MONSTER_NOT_FOUND));
+ pushBoolean(L, false);
+ return 0;
+ }
+
+ CombatType_t element = getNumber(L, 2);
+ monster->addReflectElement(element, getNumber(L, 3));
+ pushBoolean(L, true);
+ return 1;
+}
+
+int MonsterFunctions::luaMonsterAddDefense(lua_State* L) {
+ // monster:addDefense(defense)
+ const auto &monster = getUserdataShared(L, 1);
+ if (!monster) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_MONSTER_NOT_FOUND));
+ pushBoolean(L, false);
+ return 0;
+ }
+
+ monster->addDefense(getNumber(L, 2));
+ pushBoolean(L, true);
+ return 1;
+}
+
+int MonsterFunctions::luaMonsterGetDefense(lua_State* L) {
+ // monster:getDefense(defense)
+ const auto &monster = getUserdataShared(L, 1);
+ if (!monster) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_MONSTER_NOT_FOUND));
+ pushBoolean(L, false);
+ return 0;
+ }
+
+ lua_pushnumber(L, monster->getDefense());
+ return 1;
+}
+
+int MonsterFunctions::luaMonsterIsDead(lua_State* L) {
+ // monster:isDead()
+ const auto &monster = getUserdataShared(L, 1);
+ if (!monster) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_MONSTER_NOT_FOUND));
+ pushBoolean(L, false);
+ return 0;
+ }
+
+ pushBoolean(L, monster->isDead());
+ return 1;
+}
+
+int MonsterFunctions::luaMonsterImmune(lua_State* L) {
+ // to get: isImmune = monster:immune()
+ // to set and get: newImmuneBool = monster:immune(newImmuneBool)
+ const auto &monster = getUserdataShared(L, 1);
+ if (!monster) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_MONSTER_NOT_FOUND));
+ pushBoolean(L, false);
+ return 0;
+ }
+
+ if (lua_gettop(L) > 1) {
+ monster->setImmune(getBoolean(L, 2));
+ }
+
+ pushBoolean(L, monster->isImmune());
+ return 1;
+}
diff --git a/src/lua/functions/creatures/monster/monster_functions.hpp b/src/lua/functions/creatures/monster/monster_functions.hpp
index dd1c3827344..90f2b526ca4 100644
--- a/src/lua/functions/creatures/monster/monster_functions.hpp
+++ b/src/lua/functions/creatures/monster/monster_functions.hpp
@@ -64,6 +64,13 @@ class MonsterFunctions final : LuaScriptInterface {
registerMethod(L, "Monster", "hazardDamageBoost", MonsterFunctions::luaMonsterHazardDamageBoost);
registerMethod(L, "Monster", "hazardDefenseBoost", MonsterFunctions::luaMonsterHazardDefenseBoost);
+ registerMethod(L, "Monster", "addReflectElement", MonsterFunctions::luaMonsterAddReflectElement);
+ registerMethod(L, "Monster", "addDefense", MonsterFunctions::luaMonsterAddDefense);
+ registerMethod(L, "Monster", "getDefense", MonsterFunctions::luaMonsterGetDefense);
+
+ registerMethod(L, "Monster", "isDead", MonsterFunctions::luaMonsterIsDead);
+ registerMethod(L, "Monster", "immune", MonsterFunctions::luaMonsterImmune);
+
CharmFunctions::init(L);
LootFunctions::init(L);
MonsterSpellFunctions::init(L);
@@ -124,6 +131,12 @@ class MonsterFunctions final : LuaScriptInterface {
static int luaMonsterHazardDodge(lua_State* L);
static int luaMonsterHazardDamageBoost(lua_State* L);
static int luaMonsterHazardDefenseBoost(lua_State* L);
+ static int luaMonsterAddReflectElement(lua_State* L);
+ static int luaMonsterAddDefense(lua_State* L);
+ static int luaMonsterGetDefense(lua_State* L);
+
+ static int luaMonsterIsDead(lua_State* L);
+ static int luaMonsterImmune(lua_State* L);
friend class CreatureFunctions;
};
diff --git a/src/lua/functions/creatures/monster/monster_type_functions.cpp b/src/lua/functions/creatures/monster/monster_type_functions.cpp
index 757337bfe3c..2a6e3d1b581 100644
--- a/src/lua/functions/creatures/monster/monster_type_functions.cpp
+++ b/src/lua/functions/creatures/monster/monster_type_functions.cpp
@@ -1055,6 +1055,8 @@ int MonsterTypeFunctions::luaMonsterTypeEventOnCallback(lua_State* L) {
// monsterType:onDisappear(callback)
// monsterType:onMove(callback)
// monsterType:onSay(callback)
+ // monsterType:onPlayerAttack(callback)
+ // monsterType:onSpawn(callback)
const auto monsterType = getUserdataShared(L, 1);
if (monsterType) {
if (monsterType->loadCallback(&g_scripts().getScriptInterface())) {
@@ -1602,7 +1604,7 @@ int MonsterTypeFunctions::luaMonsterTypeBossRaceId(lua_State* L) {
} else {
auto raceId = getNumber(L, 2, 0);
monsterType->info.raceid = raceId;
- g_ioBosstiary().addBosstiaryMonster(raceId, monsterType->name);
+ g_ioBosstiary().addBosstiaryMonster(raceId, monsterType->typeName);
pushBoolean(L, true);
}
diff --git a/src/lua/functions/creatures/monster/monster_type_functions.hpp b/src/lua/functions/creatures/monster/monster_type_functions.hpp
index cdbd0c4158a..6923740db41 100644
--- a/src/lua/functions/creatures/monster/monster_type_functions.hpp
+++ b/src/lua/functions/creatures/monster/monster_type_functions.hpp
@@ -98,6 +98,8 @@ class MonsterTypeFunctions final : LuaScriptInterface {
registerMethod(L, "MonsterType", "onDisappear", MonsterTypeFunctions::luaMonsterTypeEventOnCallback);
registerMethod(L, "MonsterType", "onMove", MonsterTypeFunctions::luaMonsterTypeEventOnCallback);
registerMethod(L, "MonsterType", "onSay", MonsterTypeFunctions::luaMonsterTypeEventOnCallback);
+ registerMethod(L, "MonsterType", "onPlayerAttack", MonsterTypeFunctions::luaMonsterTypeEventOnCallback);
+ registerMethod(L, "MonsterType", "onSpawn", MonsterTypeFunctions::luaMonsterTypeEventOnCallback);
registerMethod(L, "MonsterType", "getSummonList", MonsterTypeFunctions::luaMonsterTypeGetSummonList);
registerMethod(L, "MonsterType", "addSummon", MonsterTypeFunctions::luaMonsterTypeAddSummon);
diff --git a/src/lua/functions/creatures/player/player_functions.cpp b/src/lua/functions/creatures/player/player_functions.cpp
index b5e207ebb10..1720a3d917c 100644
--- a/src/lua/functions/creatures/player/player_functions.cpp
+++ b/src/lua/functions/creatures/player/player_functions.cpp
@@ -2272,10 +2272,23 @@ int PlayerFunctions::luaPlayerGetParty(lua_State* L) {
}
int PlayerFunctions::luaPlayerAddOutfit(lua_State* L) {
- // player:addOutfit(lookType)
+ // player:addOutfit(lookType or name, addon = 0)
std::shared_ptr player = getUserdataShared(L, 1);
if (player) {
- player->addOutfit(getNumber(L, 2), 0);
+ auto addon = getNumber(L, 3, 0);
+ if (lua_isnumber(L, 2)) {
+ player->addOutfit(getNumber(L, 2), addon);
+ } else if (lua_isstring(L, 2)) {
+ const std::string &outfitName = getString(L, 2);
+ const auto &outfit = Outfits::getInstance().getOutfitByName(player->getSex(), outfitName);
+ if (!outfit) {
+ reportErrorFunc("Outfit not found");
+ return 1;
+ }
+
+ player->addOutfit(outfit->lookType, addon);
+ }
+
pushBoolean(L, true);
} else {
lua_pushnil(L);
@@ -4412,3 +4425,35 @@ int PlayerFunctions::luaPlayerSendIconBakragore(lua_State* L) {
pushBoolean(L, true);
return 1;
}
+
+int PlayerFunctions::luaPlayerRemoveIconBakragore(lua_State* L) {
+ // player:removeIconBakragore(iconType or nil for remove all bakragore icons)
+ const auto &player = getUserdataShared(L, 1);
+ if (!player) {
+ lua_pushnil(L);
+ return 1;
+ }
+
+ auto iconType = getNumber(L, 2, IconBakragore::None);
+ if (iconType == IconBakragore::None) {
+ player->removeBakragoreIcons();
+ } else {
+ player->removeBakragoreIcon(iconType);
+ }
+
+ pushBoolean(L, true);
+ return 1;
+}
+
+int PlayerFunctions::luaPlayerSendCreatureAppear(lua_State* L) {
+ auto player = getUserdataShared(L, 1);
+ if (!player) {
+ reportErrorFunc(getErrorDesc(LUA_ERROR_PLAYER_NOT_FOUND));
+ return 1;
+ }
+
+ bool isLogin = getBoolean(L, 2, false);
+ player->sendCreatureAppear(player, player->getPosition(), isLogin);
+ pushBoolean(L, true);
+ return 1;
+}
diff --git a/src/lua/functions/creatures/player/player_functions.hpp b/src/lua/functions/creatures/player/player_functions.hpp
index 000c5f755c0..aa4db5857f7 100644
--- a/src/lua/functions/creatures/player/player_functions.hpp
+++ b/src/lua/functions/creatures/player/player_functions.hpp
@@ -376,8 +376,9 @@ class PlayerFunctions final : LuaScriptInterface {
registerMethod(L, "Player", "createTransactionSummary", PlayerFunctions::luaPlayerCreateTransactionSummary);
registerMethod(L, "Player", "takeScreenshot", PlayerFunctions::luaPlayerTakeScreenshot);
-
registerMethod(L, "Player", "sendIconBakragore", PlayerFunctions::luaPlayerSendIconBakragore);
+ registerMethod(L, "Player", "removeIconBakragore", PlayerFunctions::luaPlayerRemoveIconBakragore);
+ registerMethod(L, "Player", "sendCreatureAppear", PlayerFunctions::luaPlayerSendCreatureAppear);
GroupFunctions::init(L);
GuildFunctions::init(L);
@@ -743,6 +744,9 @@ class PlayerFunctions final : LuaScriptInterface {
static int luaPlayerTakeScreenshot(lua_State* L);
static int luaPlayerSendIconBakragore(lua_State* L);
+ static int luaPlayerRemoveIconBakragore(lua_State* L);
+
+ static int luaPlayerSendCreatureAppear(lua_State* L);
friend class CreatureFunctions;
};
diff --git a/src/lua/functions/events/event_callback_functions.cpp b/src/lua/functions/events/event_callback_functions.cpp
index 54457f4367a..ef1c348c113 100644
--- a/src/lua/functions/events/event_callback_functions.cpp
+++ b/src/lua/functions/events/event_callback_functions.cpp
@@ -34,7 +34,14 @@ void EventCallbackFunctions::init(lua_State* luaState) {
}
int EventCallbackFunctions::luaEventCallbackCreate(lua_State* luaState) {
- const auto eventCallback = std::make_shared(getScriptEnv()->getScriptInterface());
+ const auto &callbackName = getString(luaState, 2);
+ if (callbackName.empty()) {
+ reportErrorFunc("Invalid callback name");
+ return 1;
+ }
+
+ bool skipDuplicationCheck = getBoolean(luaState, 3, false);
+ const auto eventCallback = std::make_shared(getScriptEnv()->getScriptInterface(), callbackName, skipDuplicationCheck);
pushUserdata(luaState, eventCallback);
setMetatable(luaState, -1, "EventCallback");
return 1;
@@ -82,6 +89,11 @@ int EventCallbackFunctions::luaEventCallbackRegister(lua_State* luaState) {
return 0;
}
+ if (g_callbacks().isCallbackRegistered(callback)) {
+ reportErrorFunc(fmt::format("EventCallback is duplicated for event with name: {}", callback->getName()));
+ return 0;
+ }
+
g_callbacks().addCallback(callback);
pushBoolean(luaState, true);
return 1;
diff --git a/src/lua/functions/items/container_functions.cpp b/src/lua/functions/items/container_functions.cpp
index a2251de641c..30eb26c6801 100644
--- a/src/lua/functions/items/container_functions.cpp
+++ b/src/lua/functions/items/container_functions.cpp
@@ -166,6 +166,7 @@ int ContainerFunctions::luaContainerAddItem(lua_State* L) {
setItemMetatable(L, -1, item);
} else {
reportErrorFunc(fmt::format("Cannot add item to container, error code: '{}'", getReturnMessage(ret)));
+ pushBoolean(L, false);
}
return 1;
}
diff --git a/src/map/house/house.cpp b/src/map/house/house.cpp
index 261147fff40..f2cb26114bf 100644
--- a/src/map/house/house.cpp
+++ b/src/map/house/house.cpp
@@ -871,7 +871,7 @@ void Houses::payHouses(RentPeriod_t rentPeriod) const {
}
std::ostringstream ss;
- ss << "Warning! \nThe " << period << " rent of " << house->getRent() << " gold for your house \"" << house->getName() << "\" is payable. Have it within " << daysLeft << " days or you will lose static_self_cast() house.";
+ ss << "Warning! \nThe " << period << " rent of " << house->getRent() << " gold for your house \"" << house->getName() << "\" is payable. Have it within " << daysLeft << " days or you will lose this house.";
letter->setAttribute(ItemAttribute_t::TEXT, ss.str());
g_game().internalAddItem(player->getInbox(), letter, INDEX_WHEREEVER, FLAG_NOLIMIT);
house->setPayRentWarnings(house->getPayRentWarnings() + 1);
diff --git a/src/map/map.cpp b/src/map/map.cpp
index 85e7de50484..b1706be0759 100644
--- a/src/map/map.cpp
+++ b/src/map/map.cpp
@@ -12,6 +12,8 @@
#include "map.hpp"
#include "utils/astarnodes.hpp"
+#include "lua/callbacks/event_callback.hpp"
+#include "lua/callbacks/events_callbacks.hpp"
#include "creatures/monsters/monster.hpp"
#include "game/game.hpp"
#include "game/zones/zone.hpp"
@@ -25,11 +27,7 @@ void Map::load(const std::string &identifier, const Position &pos) {
path = identifier;
IOMap::loadMap(this, pos);
} catch (const std::exception &e) {
- throw IOMapException(fmt::format(
- "\n[Map::load] - The map in folder {} is missing or corrupted"
- "\n - {}",
- identifier, e.what()
- ));
+ g_logger().warn("[Map::load] - The map in folder {} is missing or corrupted", identifier);
}
}
@@ -97,6 +95,10 @@ void Map::loadMap(const std::string &identifier, bool mainMap /*= false*/, bool
housefile.clear();
npcfile.clear();
}
+
+ if (!mainMap) {
+ g_callbacks().executeCallback(EventCallback_t::mapOnLoad, &EventCallback::mapOnLoad, path.string());
+ }
}
void Map::loadMapCustom(const std::string &mapName, bool loadHouses, bool loadMonsters, bool loadNpcs, bool loadZones, int customMapIndex) {
@@ -192,8 +194,7 @@ std::shared_ptr Map::getTile(uint16_t x, uint16_t y, uint8_t z) {
return nullptr;
}
- const auto tile = floor->getTile(x, y);
- return tile ? tile : getOrCreateTileFromCache(floor, x, y);
+ return getOrCreateTileFromCache(floor, x, y);
}
void Map::refreshZones(uint16_t x, uint16_t y, uint8_t z) {
diff --git a/src/map/mapcache.cpp b/src/map/mapcache.cpp
index e6e2f79b806..f4cdb524ec9 100644
--- a/src/map/mapcache.cpp
+++ b/src/map/mapcache.cpp
@@ -104,8 +104,9 @@ std::shared_ptr
- MapCache::createItem(const std::shared_ptr &Bas
std::shared_ptr MapCache::getOrCreateTileFromCache(const std::unique_ptr &floor, uint16_t x, uint16_t y) {
const auto cachedTile = floor->getTileCache(x, y);
+ const auto oldTile = floor->getTile(x, y);
if (!cachedTile) {
- return floor->getTile(x, y);
+ return oldTile;
}
std::unique_lock l(floor->getMutex());
@@ -114,6 +115,15 @@ std::shared_ptr MapCache::getOrCreateTileFromCache(const std::unique_ptr(this);
+ std::vector> oldCreatureList;
+ if (oldTile) {
+ if (CreatureVector* creatures = oldTile->getCreatures()) {
+ for (const auto &creature : *creatures) {
+ oldCreatureList.emplace_back(creature);
+ }
+ }
+ }
+
std::shared_ptr tile = nullptr;
if (cachedTile->isHouse()) {
const auto house = map->houses.getHouse(cachedTile->houseId);
@@ -127,6 +137,10 @@ std::shared_ptr MapCache::getOrCreateTileFromCache(const std::unique_ptrinternalAddThing(creature);
+ }
+
if (cachedTile->ground != nullptr) {
tile->internalAddThing(createItem(cachedTile->ground, pos));
}