From d4b056f4faea914699312cf77b250cf4d783053d Mon Sep 17 00:00:00 2001
From: dg <dg@51575b47-30f0-44d4-a5cc-537603b46e54>
Date: Wed, 23 Dec 2009 11:28:06 +0000
Subject: [PATCH] dumb talented AI

git-svn-id: http://svn.net-core.org/repos/t-engine4@141 51575b47-30f0-44d4-a5cc-537603b46e54
---
 game/engine/Zone.lua                          |  7 ++++-
 game/engine/ai/simple.lua                     |  7 +++--
 game/engine/ai/talented.lua                   | 28 +++++++++++++++++
 game/engine/interface/ActorTalents.lua        | 18 +++++++----
 game/modules/tome/class/Actor.lua             | 30 ++++++++++++-------
 game/modules/tome/class/NPC.lua               |  6 ++++
 game/modules/tome/class/Player.lua            |  5 ----
 game/modules/tome/data/talents/spells/air.lua |  5 ++--
 .../tome/data/talents/spells/arcane.lua       | 13 ++++----
 .../tome/data/talents/spells/conveyance.lua   | 10 +++----
 .../tome/data/talents/spells/earth.lua        |  9 +++---
 .../modules/tome/data/talents/spells/fire.lua | 17 ++++++-----
 .../tome/data/talents/spells/nature.lua       |  4 +--
 .../tome/data/talents/spells/water.lua        | 11 +++----
 .../tome/data/zones/ancient_ruins/npcs.lua    | 10 +++++--
 .../tome/data/zones/ancient_ruins/zone.lua    |  2 +-
 16 files changed, 122 insertions(+), 60 deletions(-)
 create mode 100644 game/engine/ai/talented.lua

diff --git a/game/engine/Zone.lua b/game/engine/Zone.lua
index 575f2971fd..4a47c09ad5 100644
--- a/game/engine/Zone.lua
+++ b/game/engine/Zone.lua
@@ -52,7 +52,7 @@ function _M:computeRarities(type, list, level, ood, filter)
 			elseif lev > e.level_range[2] then max = 100 / (lev - e.level_range[2])
 			end
 			local genprob = max / e.rarity
-			print("prob", e.name, math.floor(genprob), "max", math.floor(max), e.level_range[1], e.level_range[2], lev, "egoable", e.egos and #e.egos)
+			print("prob", e.name, math.floor(genprob), "max", math.floor(max), e.level_range[1], e.level_range[2], lev, "egoable", e.egos)
 
 			-- Generate and store egos list if needed
 			if e.egos and not level:getEntitiesList(type.."/"..e.egos) then
@@ -68,7 +68,12 @@ function _M:computeRarities(type, list, level, ood, filter)
 		end
 	end
 	table.sort(r, function(a, b) return a.genprob < b.genprob end)
+
 	print("*DONE", r.total)
+	for i, ee in ipairs(r) do
+		print(("entity chance %02d : chance(%04d): %s"):format(i, ee.genprob, ee.e.name))
+	end
+
 	return r
 end
 
diff --git a/game/engine/ai/simple.lua b/game/engine/ai/simple.lua
index 2024eaa61c..62a06374ea 100644
--- a/game/engine/ai/simple.lua
+++ b/game/engine/ai/simple.lua
@@ -10,7 +10,7 @@ newAI("move_simple", function(self)
 end)
 
 newAI("target_simple", function(self)
-	if self.ai_target.actor and not self.ai_target.actor.dead and rng.percent(90) then return end
+	if self.ai_target.actor and not self.ai_target.actor.dead and rng.percent(90) then return true end
 
 	-- Find closer ennemy and target it
 	-- Get list of actors ordered by distance
@@ -40,6 +40,7 @@ newAI("target_player", function(self)
 end)
 
 newAI("simple", function(self)
-	self:runAI("target_simple")
-	self:runAI("move_simple")
+	if self:runAI("target_simple") then
+		self:runAI("move_simple")
+	end
 end)
diff --git a/game/engine/ai/talented.lua b/game/engine/ai/talented.lua
new file mode 100644
index 0000000000..7cd2765281
--- /dev/null
+++ b/game/engine/ai/talented.lua
@@ -0,0 +1,28 @@
+-- Defines AIs that can use talents, either smartly or "dumbly"
+
+-- Randomly use talents
+newAI("dumb_talented", function(self)
+	-- Find available talents
+	local avail = {}
+	local target_dist = core.fov.distance(self.x, self.y, self.ai_target.actor.x, self.ai_target.actor.y)
+	for i, tid in ipairs(self.talents) do
+		local t = self:getTalentFromId(tid)
+		if not self:isTalentCoolingDown(t) and target_dist <= self:getTalentRange(t) and self:preUseTalent(t, true) then
+			avail[#avail+1] = tid
+			print(self.name, self.uid, "dumb ai talents can use", t.name)
+		end
+	end
+	if #avail > 0 then
+		local tid = avail[rng.range(1, #avail)]
+		self:useTalent(tid)
+	end
+end)
+
+newAI("dumb_talented_simple", function(self)
+	if self:runAI("target_simple") then
+		-- One in "talent_in" chance of using a talent
+		if rng.chance(self.ai_state.talent_in or 6) and not self:runAI("dumb_talented") then
+			self:runAI("move_simple")
+		end
+	end
+end)
diff --git a/game/engine/interface/ActorTalents.lua b/game/engine/interface/ActorTalents.lua
index dd4b6c5042..7a63e5e5ae 100644
--- a/game/engine/interface/ActorTalents.lua
+++ b/game/engine/interface/ActorTalents.lua
@@ -83,7 +83,7 @@ function _M:useTalent(id)
 		end
 		if not self:preUseTalent(ab) then return end
 		local co = coroutine.create(function()
-			local ret = ab.action(self)
+			local ret = ab.action(self, ab)
 
 			if not self:postUseTalent(ab, ret) then return end
 
@@ -100,13 +100,13 @@ function _M:useTalent(id)
 		if not self:preUseTalent(ab) then return end
 		local co = coroutine.create(function()
 			if not self.sustain_talents[id] then
-				local ret = ab.activate(self)
+				local ret = ab.activate(self, ab)
 
 				if not self:postUseTalent(ab, ret) then return end
 
 				self.sustain_talents[id] = ret
 			else
-				local ret = ab.deactivate(self, self.sustain_talents[id])
+				local ret = ab.deactivate(self, ab, self.sustain_talents[id])
 
 				if not self:postUseTalent(ab, ret) then return end
 
@@ -120,6 +120,7 @@ function _M:useTalent(id)
 	else
 		error("Activating non activable or sustainable talent: "..id.." :: "..ab.name.." :: "..ab.mode)
 	end
+	return true
 end
 
 --- Replace some markers in a string with info on the talent
@@ -183,7 +184,7 @@ function _M:learnTalent(t_id, force)
 		end
 	end
 
-	if t.on_learn then t.on_learn(self) end
+	if t.on_learn then t.on_learn(self, t) end
 
 	self.talents[t_id] = true
 	self.changed = true
@@ -200,7 +201,7 @@ function _M:unlearnTalent(t_id)
 		if known_t_id == t_id then self.hotkey[i] = nil end
 	end
 
-	if t.on_unlearn then t.on_unlearn(self) end
+	if t.on_unlearn then t.on_unlearn(self, t) end
 
 	self.talents[t_id] = nil
 	self.changed = true
@@ -325,6 +326,13 @@ function _M:isTalentCoolingDown(t)
 	if self.talents_cd[t.id] and self.talents_cd[t.id] > 0 then return self.talents_cd[t.id] else return false end
 end
 
+--- Returns the range of a talent
+function _M:getTalentRange(t)
+	if not t.range then return 1 end
+	if type(t.range) == "function" then return t.range(self, t) end
+	return t.range
+end
+
 --- Cooldown all talents by one
 -- This should be called in your actors "act()" method
 function _M:cooldownTalents()
diff --git a/game/modules/tome/class/Actor.lua b/game/modules/tome/class/Actor.lua
index 27646543ce..a9cac9e5ae 100644
--- a/game/modules/tome/class/Actor.lua
+++ b/game/modules/tome/class/Actor.lua
@@ -51,6 +51,12 @@ function _M:init(t, no_default)
 	t.mana_rating = t.mana_rating or 10
 	t.stamina_rating = t.stamina_rating or 10
 
+	-- Default regen
+	self.mana_regen = 0.5
+	self.stamina_regen = 0.5
+	self.life_regen = 0.1
+	self.life_rating = 10
+
 	-- Default melee barehanded damage
 	self.combat = { dam=1, atk=1, apr=0, dammod={str=1} }
 
@@ -183,7 +189,7 @@ end
 -- Check the actor can cast it
 -- @param ab the talent (not the id, the table)
 -- @return true to continue, false to stop
-function _M:preUseTalent(ab)
+function _M:preUseTalent(ab, silent)
 	if not self:enoughEnergy() then return end
 	if ab.mana and self:getMana() < ab.mana * (100 + self.fatigue) / 100 then
 		game.logPlayer(self, "You do not have enough mana to cast %s.", ab.name)
@@ -194,16 +200,18 @@ function _M:preUseTalent(ab)
 		return
 	end
 
-	if ab.message then
-		game.logSeen(self, "%s", self:useTalentMessage(ab))
-	elseif ab.mode == "sustained" and not self:isTalentActive(ab.id) then
-		game.logSeen(self, "%s activates %s.", self.name:capitalize(), ab.name)
-	elseif ab.mode == "sustained" and self:isTalentActive(ab.id) then
-		game.logSeen(self, "%s deactivates %s.", self.name:capitalize(), ab.name)
-	elseif ab.type[1]:find("^spell/") then
-		game.logSeen(self, "%s casts %s.", self.name:capitalize(), ab.name)
-	else
-		game.logSeen(self, "%s uses %s.", self.name:capitalize(), ab.name)
+	if not silent then
+		if ab.message then
+			game.logSeen(self, "%s", self:useTalentMessage(ab))
+		elseif ab.mode == "sustained" and not self:isTalentActive(ab.id) then
+			game.logSeen(self, "%s activates %s.", self.name:capitalize(), ab.name)
+		elseif ab.mode == "sustained" and self:isTalentActive(ab.id) then
+			game.logSeen(self, "%s deactivates %s.", self.name:capitalize(), ab.name)
+		elseif ab.type[1]:find("^spell/") then
+			game.logSeen(self, "%s casts %s.", self.name:capitalize(), ab.name)
+		else
+			game.logSeen(self, "%s uses %s.", self.name:capitalize(), ab.name)
+		end
 	end
 	return true
 end
diff --git a/game/modules/tome/class/NPC.lua b/game/modules/tome/class/NPC.lua
index 47530eec3c..eb5108774e 100644
--- a/game/modules/tome/class/NPC.lua
+++ b/game/modules/tome/class/NPC.lua
@@ -30,3 +30,9 @@ function _M:tooltip()
 	local str = mod.class.Actor.tooltip(self)
 	return str..("\nTarget: %s\nUID: %d"):format(self.ai_target.actor and self.ai_target.actor.name or "none", self.uid)
 end
+
+--- Tries to get a target from the NPC
+-- This simple returns current AI target for NPCs
+function _M:getTarget(typ)
+	return self.ai_target.actor.x, self.ai_target.actor.y
+end
diff --git a/game/modules/tome/class/Player.lua b/game/modules/tome/class/Player.lua
index c739697f3f..2aa825dff6 100644
--- a/game/modules/tome/class/Player.lua
+++ b/game/modules/tome/class/Player.lua
@@ -30,11 +30,6 @@ function _M:init(t, no_default)
 	self.color_b=230
 --	self.image="player.png"
 
-	-- Default regen
-	self.mana_regen = 0.5
-	self.stamina_regen = 0.5
-	self.life_regen = 0.1
-	self.life_rating = 10
 	self.fixed_rating = true
 
 	self.max_life=85
diff --git a/game/modules/tome/data/talents/spells/air.lua b/game/modules/tome/data/talents/spells/air.lua
index 0e02a5bbd6..d7067e6b3b 100644
--- a/game/modules/tome/data/talents/spells/air.lua
+++ b/game/modules/tome/data/talents/spells/air.lua
@@ -6,11 +6,12 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 10,
 	},
-	action = function(self)
+	range = 15,
+	action = function(self, t)
 		local duration = 5 + self:combatSpellpower(0.1)
 		local radius = 3
 		local dam = 4 + self:combatSpellpower(0.6)
-		local t = {type="ball", range=15, radius=radius}
+		local t = {type="ball", range=self:getTalentRange(t), radius=radius}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		x, y = game.target:pointAtRange(self.x, self.y, x, y, 15)
diff --git a/game/modules/tome/data/talents/spells/arcane.lua b/game/modules/tome/data/talents/spells/arcane.lua
index 8730e07340..6404793f4a 100644
--- a/game/modules/tome/data/talents/spells/arcane.lua
+++ b/game/modules/tome/data/talents/spells/arcane.lua
@@ -6,8 +6,9 @@ newTalent{
 	tactical = {
 		ATTACK = 10,
 	},
-	action = function(self)
-		local t = {type="bolt", range=20}
+	range = 20,
+	action = function(self, t)
+		local t = {type="bolt", range=self:getTalentRange(t)}
 		if self:knowTalent(Talents.T_ARCANE_LANCE) then t.type = "beam" end
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
@@ -38,7 +39,7 @@ newTalent{
 	tactical = {
 		MANA = 20,
 	},
-	action = function(self)
+	action = function(self, t)
 		if not self:hasEffect(self.EFF_MANAFLOW) then
 			self:setEffect(self.EFF_MANAFLOW, 10, {power=5+self:combatSpellpower(0.3)})
 		end
@@ -56,10 +57,10 @@ newTalent{
 	type = {"spell/arcane", 3},
 	mode = "passive",
 	require = { stat = { mag=40 }, },
-	on_learn = function(self)
+	on_learn = function(self, t)
 		self.combat_spellpower = self.combat_spellpower + 10
 	end,
-	on_unlearn = function(self)
+	on_unlearn = function(self, t)
 		self.combat_spellpower = self.combat_spellpower - 10
 	end,
 	info = function(self)
@@ -75,7 +76,7 @@ newTalent{
 	tactical = {
 		DEFEND = 10,
 	},
-	activate = function(self)
+	activate = function(self, t)
 		game.log("IMPLEMENT ME!")
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/spells/conveyance.lua b/game/modules/tome/data/talents/spells/conveyance.lua
index 0521bf433e..21404ac7d1 100644
--- a/game/modules/tome/data/talents/spells/conveyance.lua
+++ b/game/modules/tome/data/talents/spells/conveyance.lua
@@ -7,7 +7,7 @@ newTalent{
 	tactical = {
 		ESCAPE = 4,
 	},
-	action = function(self)
+	action = function(self, t)
 		local target = self
 
 		if self:knowTalent(Talents.T_TARGETED_TELEPORT) then
@@ -46,7 +46,7 @@ newTalent{
 	tactical = {
 		ESCAPE = 8,
 	},
-	action = function(self)
+	action = function(self, t)
 		local target = self
 
 		if self:knowTalent(Talents.T_TARGETED_TELEPORT) then
@@ -108,11 +108,11 @@ newTalent{
 	tactical = {
 		MOVEMENT = 20,
 	},
-	activate = function(self)
+	activate = function(self, t)
 		self:attr("prob_travel", 1)
 		return true
 	end,
-	deactivate = function(self)
+	deactivate = function(self, t)
 		self:attr("prob_travel", -1)
 		return true
 	end,
@@ -127,7 +127,7 @@ newTalent{
 	type = {"spell/conveyance",3},
 	mana = 30,
 	cooldown = 10,
-	action = function(self)
+	action = function(self, t)
 --[[
 		local target = self
 		local tx, ty = self.x, self.y
diff --git a/game/modules/tome/data/talents/spells/earth.lua b/game/modules/tome/data/talents/spells/earth.lua
index e8e835b63a..51a3a73fe2 100644
--- a/game/modules/tome/data/talents/spells/earth.lua
+++ b/game/modules/tome/data/talents/spells/earth.lua
@@ -7,12 +7,12 @@ newTalent{
 	tactical = {
 		DEFEND = 10,
 	},
-	activate = function(self)
+	activate = function(self, t)
 		local power = 1 + self:combatSpellpower(0.15)
 		self.combat.armor = self.combat.armor + power
 		return {power=power}
 	end,
-	deactivate = function(self, p)
+	deactivate = function(self, t, p)
 		self.combat.armor = self.combat.armor - p.power
 	end,
 	require = { stat = { mag=14 }, },
@@ -30,8 +30,9 @@ newTalent{
 	tactical = {
 		ATTACK = 10,
 	},
-	action = function(self)
-		local t = {type="bolt", range=20}
+	range = 20,
+	action = function(self, t)
+		local t = {type="bolt", range=self:getTalentRange(t)}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		self:project(t, x, y, DamageType.SPELLKNOCKBACK, self:spellCrit(8 + self:combatSpellpower(0.6)))
diff --git a/game/modules/tome/data/talents/spells/fire.lua b/game/modules/tome/data/talents/spells/fire.lua
index da376bfbcf..6da8ad7842 100644
--- a/game/modules/tome/data/talents/spells/fire.lua
+++ b/game/modules/tome/data/talents/spells/fire.lua
@@ -6,7 +6,7 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 3,
 	},
-	action = function(self)
+	action = function(self, t)
 		local t = {type="ball", range=0, friendlyfire=false, radius=5 + self:combatSpellpower(0.2)}
 		self:project(t, self.x, self.y, DamageType.LIGHT, 1)
 		if self:knowTalent(Talents.T_GLOBE_OF_LIGHT) then
@@ -38,8 +38,9 @@ newTalent{
 	tactical = {
 		ATTACK = 10,
 	},
-	action = function(self)
-		local t = {type="bolt", range=20}
+	range = 20,
+	action = function(self, t)
+		local t = {type="bolt", range=self:getTalentRange(t)}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		self:project(t, x, y, DamageType.FIREBURN, self:spellCrit(15 + self:combatSpellpower(2.1)))
@@ -60,8 +61,9 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 10,
 	},
-	action = function(self)
-		local t = {type="ball", range=15, radius=math.min(6, 3 + self:combatSpellpower(0.06))}
+	range = 15,
+	action = function(self, t)
+		local t = {type="ball", range=self:getTalentRange(t), radius=math.min(6, 3 + self:combatSpellpower(0.06))}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		self:project(t, x, y, DamageType.FIRE, self:spellCrit(28 + self:combatSpellpower(1.2)))
@@ -82,11 +84,12 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 40,
 	},
-	action = function(self)
+	range = 20,
+	action = function(self, t)
 		local duration = 5 + self:combatSpellpower(0.25)
 		local radius = 5
 		local dam = 15 + self:combatSpellpower(1.6)
-		local t = {type="ball", range=20, radius=radius}
+		local t = {type="ball", range=self:getTalentRange(t), radius=radius}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		x, y = game.target:pointAtRange(self.x, self.y, x, y, 15)
diff --git a/game/modules/tome/data/talents/spells/nature.lua b/game/modules/tome/data/talents/spells/nature.lua
index 7a86230d7d..9b9248f3b7 100644
--- a/game/modules/tome/data/talents/spells/nature.lua
+++ b/game/modules/tome/data/talents/spells/nature.lua
@@ -6,7 +6,7 @@ newTalent{
 	tactical = {
 		HEAL = 10,
 	},
-	action = function(self)
+	action = function(self, t)
 		self:setEffect(self.EFF_REGENERATION, 10, {power=5 + self:combatSpellpower(0.35)})
 		return true
 	end,
@@ -25,7 +25,7 @@ newTalent{
 	tactical = {
 		HEAL = 10,
 	},
-	action = function(self)
+	action = function(self, t)
 		self:heal(self:spellCrit(10 + self:combatSpellpower(2)), self)
 		return true
 	end,
diff --git a/game/modules/tome/data/talents/spells/water.lua b/game/modules/tome/data/talents/spells/water.lua
index 54b4724cf5..3428d96792 100644
--- a/game/modules/tome/data/talents/spells/water.lua
+++ b/game/modules/tome/data/talents/spells/water.lua
@@ -3,7 +3,7 @@ newTalent{
 	type = {"spell/water", 1},
 	mana = 10,
 	cooldown = 100,
-	action = function(self)
+	action = function(self, t)
 		return true
 	end,
 	require = { stat = { mag=10 }, },
@@ -20,8 +20,9 @@ newTalent{
 	tactical = {
 		ATTACK = 10,
 	},
-	action = function(self)
-		local t = {type="hit", range=20}
+	range = 20,
+	action = function(self, t)
+		local t = {type="hit", range=self:getTalentRange(t)}
 		local x, y = self:getTarget(t)
 		if not x or not y then return nil end
 		self:project(t, x, y, DamageType.COLD, self:spellCrit(7 + self:combatSpellpower(1.2)))
@@ -43,7 +44,7 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 10,
 	},
-	action = function(self)
+	action = function(self, t)
 		local duration = 5 + self:combatSpellpower(0.05)
 		local radius = 1
 		local dam = 12 + self:combatSpellpower(0.5)
@@ -76,7 +77,7 @@ newTalent{
 	tactical = {
 		ATTACKAREA = 20,
 	},
-	action = function(self)
+	action = function(self, t)
 		local duration = 5 + self:combatSpellpower(0.25)
 		local radius = 3
 		local dam = 12 + self:combatSpellpower(0.8)
diff --git a/game/modules/tome/data/zones/ancient_ruins/npcs.lua b/game/modules/tome/data/zones/ancient_ruins/npcs.lua
index 0e402a1f9f..bd0988608e 100644
--- a/game/modules/tome/data/zones/ancient_ruins/npcs.lua
+++ b/game/modules/tome/data/zones/ancient_ruins/npcs.lua
@@ -1,3 +1,5 @@
+local Talents = require("engine.interface.ActorTalents")
+
 newEntity{
 	group = "dragon",
 	name = "dragon of death",
@@ -40,11 +42,13 @@ newEntity{
 	level_range = {1, 7}, exp_worth = 1,
 	rarity = 1,
 	autolevel = "warrior",
-	ai = "simple",
+	ai = "dumb_talented_simple",
 	max_life = resolvers.rngavg(10,20),
-	max_mana = resolvers.rngavg(10,20),
+	max_mana = resolvers.rngavg(50,60),
 	max_stamina = resolvers.rngavg(10,20),
 	energy = { mod=0.3 },
 	has_blood = {nb=3, color={50,255,120}},
-	combat = { dam=5, atk=6, def=2, apr=1, armor=2},
+	combat = { dam=5, atk=6, def=2, apr=1, armor=2 },
+	stats = { str=10, dex=7, mag=14, con=10 },
+	talents = { Talents.T_MANATHRUST, Talents.T_FREEZE, Talents.T_FLAME }
 }
diff --git a/game/modules/tome/data/zones/ancient_ruins/zone.lua b/game/modules/tome/data/zones/ancient_ruins/zone.lua
index 7d243d86d9..8850f21e25 100644
--- a/game/modules/tome/data/zones/ancient_ruins/zone.lua
+++ b/game/modules/tome/data/zones/ancient_ruins/zone.lua
@@ -19,7 +19,7 @@ return {
 		},
 		actor = {
 			class = "engine.generator.actor.Random",
-			nb_npc = {2, 3},
+			nb_npc = {20, 30},
 			ood = {chance=5, range={1, 10}},
 			adjust_level_to_player = {-1, 2},
 		},
-- 
GitLab