local helper = wesnoth.require "lua/helper.lua"
local wml_actions = wesnoth.wml_actions
local game_events = wesnoth.game_events
local function color_prefix(r, g, b)
return string.format('', r, g, b)
end
local function insert_before_nl(s, t)
return string.gsub(tostring(s), "[^\n]*", "%0" .. t, 1)
end
local scenario_objectives = {}
local old_on_save = game_events.on_save
function game_events.on_save()
local custom_cfg = old_on_save()
for i,v in pairs(scenario_objectives) do
v.side = i
table.insert(custom_cfg, { "objectives", v })
end
return custom_cfg
end
local old_on_load = game_events.on_load
function game_events.on_load(cfg)
for i = #cfg,1,-1 do
local v = cfg[i]
if v[1] == "objectives" then
local v2 = v[2]
scenario_objectives[v2.side or 0] = v2
table.remove(cfg, i)
end
end
old_on_load(cfg)
end
local function generate_objectives(cfg)
-- Note: when changing the text formatting, remember to check if you also
-- need to change the hardcoded default multiplayer objective text in
-- multiplayer_connect.cpp.
local _ = wesnoth.textdomain("wesnoth")
local objectives = ""
local win_objectives = ""
local lose_objectives = ""
local gold_carryover = ""
local notes = ""
local win_string = cfg.victory_string or _ "Victory:"
local lose_string = cfg.defeat_string or _ "Defeat:"
local gold_carryover_string = cfg.gold_carryover_string or _ "Gold carryover:"
local notes_string = cfg.notes_string or _ "Notes:"
local bullet = cfg.bullet or "• "
for obj in helper.child_range(cfg, "objective") do
local show_if = helper.get_child(obj, "show_if")
if not show_if or wesnoth.eval_conditional(show_if) then
local objective_bullet = obj.bullet or bullet
local condition = obj.condition
local description = obj.description or ""
local turn_counter = ""
if obj.show_turn_counter then
local current_turn = wesnoth.current.turn
local turn_limit = wesnoth.game_config.last_turn
if turn_limit >= current_turn then
if turn_limit - current_turn + 1 > 1 then
turn_counter = " " .. string.format(tostring(_"(%d turns left)"), turn_limit - current_turn + 1) .. ""
else
turn_counter = " " .. _"(this turn left)" .. ""
end
end
end
if condition == "win" then
local caption = obj.caption
local r = obj.red or 0
local g = obj.green or 255
local b = obj.blue or 0
if caption then
win_objectives = win_objectives .. caption .. "\n"
end
win_objectives = win_objectives .. color_prefix(r, g, b) .. objective_bullet .. description .. turn_counter .. "" .. "\n"
elseif condition == "lose" then
local caption = obj.caption
local r = obj.red or 255
local g = obj.green or 0
local b = obj.blue or 0
if caption then
lose_objectives = lose_objectives .. caption .. "\n"
end
lose_objectives = lose_objectives .. color_prefix(r, g, b) .. objective_bullet .. description .. turn_counter .. "" .. "\n"
else
wesnoth.message "Unknown condition, ignoring."
end
end
end
for obj in helper.child_range(cfg, "gold_carryover") do
local gold_carryover_bullet = obj.bullet or bullet
local r = obj.red or 255
local g = obj.green or 255
local b = obj.blue or 192
if obj.bonus ~= nil then
if obj.bonus then
gold_carryover = color_prefix(r, g, b) .. gold_carryover_bullet .. "" .. _"Early finish bonus." .. "\n"
else
gold_carryover = color_prefix(r, g, b) .. gold_carryover_bullet .. "" .. _"No early finish bonus." .. "\n"
end
end
if obj.carryover_percentage then
local carryover_amount_string = ""
if obj.carryover_percentage == 0 then
carryover_amount_string = _"No gold carried over to the next scenario."
else
carryover_amount_string = string.format(tostring(_ "%d%% of gold carried over to the next scenario."), obj.carryover_percentage)
end
gold_carryover = gold_carryover .. color_prefix(r, g, b) .. gold_carryover_bullet .. "" .. carryover_amount_string .. "\n"
end
end
for note in helper.child_range(cfg, "note") do
local show_if = helper.get_child(note, "show_if")
if not show_if or wesnoth.eval_conditional(show_if) then
local note_bullet = note.bullet or bullet
local r = note.red or 255
local g = note.green or 255
local b = note.blue or 255
if note.description then
notes = notes .. color_prefix(r, g, b) .. note_bullet .. "" .. note.description .. "\n"
end
end
end
local summary = cfg.summary
if summary then
objectives = "" .. insert_before_nl(summary, "") .. "\n"
end
if win_objectives ~= "" then
objectives = objectives .. "" .. win_string .. "\n" .. win_objectives
end
if lose_objectives ~= "" then
objectives = objectives .. "" .. lose_string .. "\n" .. lose_objectives
end
if gold_carryover ~= "" then
objectives = objectives .. gold_carryover_string .. "\n" .. gold_carryover
end
if notes ~= "" then
objectives = objectives .. notes_string .. "\n" .. notes
end
local note = cfg.note
if note then
objectives = objectives .. note .. "\n"
end
return string.sub(tostring(objectives), 1, -2)
end
local function remove_ssf_info_from(cfg)
cfg.side = nil
cfg.team_name = nil
for i, v in ipairs(cfg) do
if v[1] == "has_unit" or v[1] == "enemy_of" or v[1] == "allied_with" then
table.remove(cfg, i)
end
end
end
function wml_actions.objectives(cfg)
cfg = helper.parsed(cfg)
local sides = wesnoth.get_sides(cfg)
local silent = cfg.silent
remove_ssf_info_from(cfg)
cfg.silent = nil
local objectives = generate_objectives(cfg)
local function set_objectives(sides, save)
for i, team in ipairs(sides) do
if save then scenario_objectives[team.side] = cfg end
team.objectives = objectives
team.objectives_changed = not silent
end
end
if #sides == #wesnoth.sides or #sides == 0 then
scenario_objectives[0] = cfg
set_objectives(wesnoth.sides)
else
set_objectives(sides, true)
end
end
function wml_actions.show_objectives(cfg)
local cfg0 = scenario_objectives[0]
local function local_show_objectives(sides)
local objectives0 = cfg0 and generate_objectives(cfg0)
for i, team in ipairs(sides) do
cfg = scenario_objectives[team.side]
local objectives = (cfg and generate_objectives(cfg)) or objectives0
if objectives then team.objectives = objectives end
team.objectives_changed = true
end
end
local sides = wesnoth.get_sides(cfg)
if #sides == 0 then
local_show_objectives(wesnoth.sides)
else
local_show_objectives(sides)
end
end