From 19ca916d135379109bc85316710a9c04835ebcac Mon Sep 17 00:00:00 2001 From: Flibe Date: Thu, 1 Feb 2024 23:53:21 -0500 Subject: [PATCH] puzzle 3 triva challenge implementation Almost all logic for determining cutscenes is handled by event arguments I've done my best to make all the arguments self-explanatory via their names --- .../13_A_Challenge_You_Could_Be_a_Winner.lua | 125 ++++------ scripts/missions/amk/helpers.lua | 229 +++++++++++++++++- scripts/zones/Xarcabard/DefaultActions.lua | 11 +- scripts/zones/Xarcabard/IDs.lua | 13 +- scripts/zones/Xarcabard/npcs/Option_One.lua | 22 ++ scripts/zones/Xarcabard/npcs/Option_Three.lua | 21 ++ scripts/zones/Xarcabard/npcs/Option_Two.lua | 21 ++ 7 files changed, 352 insertions(+), 90 deletions(-) create mode 100644 scripts/zones/Xarcabard/npcs/Option_One.lua create mode 100644 scripts/zones/Xarcabard/npcs/Option_Three.lua create mode 100644 scripts/zones/Xarcabard/npcs/Option_Two.lua diff --git a/scripts/missions/amk/13_A_Challenge_You_Could_Be_a_Winner.lua b/scripts/missions/amk/13_A_Challenge_You_Could_Be_a_Winner.lua index f93bff51b18..2ae401623d4 100644 --- a/scripts/missions/amk/13_A_Challenge_You_Could_Be_a_Winner.lua +++ b/scripts/missions/amk/13_A_Challenge_You_Could_Be_a_Winner.lua @@ -74,7 +74,7 @@ mission.sections = player:needToZone(true) end end, - } + }, }, }, @@ -100,14 +100,14 @@ mission.sections = ['Goblin_Grenadier'] = { onTrigger = function(player, npc) - local answer = player:getLocalVar('Mission[10][12]pipSet') - 1 + local answer = player:getLocalVar('Mission[10][12][p1]pipSet') - 1 if answer < 0 then return mission:progressEvent(508, xi.ki.MAP_OF_THE_NORTHLANDS_AREA) else local today = VanadielDayOfTheWeek() local tomorrow = (today + 1) % 8 - local hintsUsed = player:getLocalVar('Mission[10][12]hintsUsed') + local hintsUsed = player:getLocalVar('Mission[10][12][p1]hintsUsed') player:messageSpecial(zones[player:getZoneID()].text.GRENADIER_DAY_HINT, today, tomorrow) @@ -126,108 +126,42 @@ mission.sections = ['Northwestern_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][1] - - return mission:progressEvent( - 510, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 1) end, }, ['Western_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][2] - - return mission:progressEvent( - 511, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 2) end, }, ['Southwestern_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][3] - - return mission:progressEvent( - 512, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 3) end, }, ['Northeastern_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][4] - - return mission:progressEvent( - 513, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 4) end, }, ['Eastern_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][5] - - return mission:progressEvent( - 514, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 5) end, }, ['Southeastern_Pip'] = { onTrigger = function(player, npc) - local pipSet = player:getLocalVar('Mission[10][12]pipSet') - 1 - local pos = npc:getPos() - local element = xi.amk.helpers.pipSets[pipSet][6] - - return mission:progressEvent( - 515, - pos.x * 1000, - pos.z * 1000, - pos.y * 1000, - element, - xi.ki.MAP_OF_THE_NORTHLANDS_AREA - ) + return xi.amk.helpers.puzzleOneOnTrigger(player, npc, mission, 6) end, }, @@ -249,8 +183,8 @@ mission.sections = option == 1 or -- Used first hint option == 2 -- Used second hint then - local hintsUsed = player:getLocalVar('Mission[10][12]hintsUsed') - player:setLocalVar('Mission[10][12]hintsUsed', hintsUsed + 1) + local hintsUsed = player:getLocalVar('Mission[10][12][p1]hintsUsed') + player:setLocalVar('Mission[10][12][p1]hintsUsed', hintsUsed + 1) elseif option == 3 then -- wrong answer, reset puzzle player:needToZone(false) @@ -280,7 +214,7 @@ mission.sections = [508] = function(player, csid, option, npc) -- Pipset offset by 1 to account for saving 0 as a variable. When retrieving, subtract 1 - player:setLocalVar('Mission[10][12]pipSet', math.random(1, 10)) -- range: 0 - 9 + player:setLocalVar('Mission[10][12][p1]pipSet', math.random(1, 10)) -- range: 0 - 9 end, }, }, @@ -325,7 +259,7 @@ mission.sections = -- Part 3: Xarcabard { check = function(player, currentMission, missionStatus, vars) - return currentMission == mission.missionId and + return currentMission >= mission.missionId and player:getCharVar('Mission[10][12]progress') == 2 and player:hasKeyItem(xi.ki.TRIVIA_CHALLENGE_KUPON) end, @@ -343,7 +277,40 @@ mission.sections = [xi.zone.XARCABARD] = { + ['Option_One'] = + { + onTrigger = function(player, npc) + return xi.amk.helpers.puzzleTwoOnTrigger(player, npc, mission) + end, + }, + + ['Option_Two'] = + { + onTrigger = function(player, npc) + return xi.amk.helpers.puzzleTwoOnTrigger(player, npc, mission) + end, + }, + + ['Option_Three'] = + { + onTrigger = function(player, npc) + return xi.amk.helpers.puzzleTwoOnTrigger(player, npc, mission) + end, + }, + + onEventUpdate = + { + [200] = function(player, csid, option, npc) + xi.amk.helpers.puzzleTwoOnEventUpdate(player, csid, option, npc, mission) + end, + }, + onEventFinish = + { + [200] = function(player, csid, option, npc) + xi.amk.helpers.puzzleTwoOnEventFinish(player, csid, option, npc, mission) + end + }, }, }, diff --git a/scripts/missions/amk/helpers.lua b/scripts/missions/amk/helpers.lua index f1131a0bd0a..f266408b95e 100644 --- a/scripts/missions/amk/helpers.lua +++ b/scripts/missions/amk/helpers.lua @@ -1,6 +1,9 @@ ----------------------------------- -- A Moogle Kupo d'Etat Helpers ----------------------------------- +XARC = zones[xi.zone.XARCABARD] +----------------------------------- + xi = xi or {} xi.amk = xi.amk or {} xi.amk.helpers = xi.amk.helpers or {} @@ -222,7 +225,7 @@ xi.amk.helpers.digSites = { x = 392.970, z = 157.610 }, -- (J-7) NE corner { x = -32.940, z = 251.870 }, -- (H-6) Green patch on ground { x = -275.140, z = -258.90 }, -- (F-9) Center of grid (kind of impossible) - { x = 426.50, z = -231.660 }, -- (K-9) Few feet west of sign post + { x = 426.50, z = -231.660 }, -- (K-9) Few feet west of randomSign post }, }, [xi.zone.MERIPHATAUD_MOUNTAINS] = @@ -417,10 +420,13 @@ xi.amk.helpers.chocoboDig = function(player, zoneId, text) end -- Mission 13 (index 12) - Puzzles!!! --- Puzzle 1 +----------------------------------- +-- Puzzle 1 - Elemental Numbers! +----------------------------------- xi.amk.helpers.pipSets = { - -- [Answer]: 0-9 { + -- [Answer]: 0-9 + -- { -- [1] = NW, [4] = NE, -- [2] = W, [5] = E, -- [3] = SW, [6] = SE, @@ -495,3 +501,220 @@ xi.amk.helpers.pipSets = [3] = 1, [6] = 3, }, } + +xi.amk.helpers.puzzleOneOnTrigger = function(player, npc, mission, offset) + local pipSet = player:getLocalVar('Mission[10][12][p1]pipSet') - 1 + local pos = npc:getPos() + local element = xi.amk.helpers.pipSets[pipSet][offset] + + return mission:progressEvent(509 + offset, + pos.x * 1000, + pos.z * 1000, + pos.y * 1000, + element, + xi.ki.MAP_OF_THE_NORTHLANDS_AREA + ) +end + +----------------------------------- +-- Puzzle 2 - Trivia! +-- CSID 200 trigger params: question number, correct answer (stoogeNum), previous stooge, stooge location, time limit, current time, player ID? +-- CSID 200 update params: answer 1, location 1, answer 2, location 2, cs moogle location, time limit, current time, player id? +-- trivia locations: 0 = option one, 1 = option two, 2 = option three +-- Stooge CS locations: 1 = option one, 2 = option two, 3 = option three +-- Variables are named as 'option' if 0-indexed, as 'stooge' if 1-indexed +-- TODO: Figure out way to add questions 4, 6, 7, 8, and 9 +------------------------ + +local randomSign = function() + -- returns -1 or 1 to offset the wrong answer randomly + return math.random(1,2) == 1 and -1 or 1 +end + +-- Structured list of the trivia questions +xi.amk.helpers.triviaQuestions = +{ + -- 0 : Add your current hit points to your current magic points, and you get…? + [0] = function(player) + local right = player:getHP() + player:getMP() + local wrong = right + (randomSign() * 10) + return {right, wrong} + end, + + -- 1 : The sum total of each and every one of your job levels is? + [1] = function(player) + local job_total = 0 + for i = xi.job.WAR, xi.job.RUN do + job_total = job_total + player:getJobLevel(i) + end + local right = job_total + local wrong = right + (randomSign() * math.random(1, 15)) + return {right, wrong} + end, + + -- 2 : The sum total of each of your crafting skill levels is? + [2] = function(player) + local craft_total = 0 + for i = xi.skill.FISHING, xi.skill.COOKING do + craft_total = craft_total + math.floor(player:getCharSkillLevel(i) / 10) + end + local right = craft_total + local wrong = right + (randomSign() * math.random(1, 10)) + if right == 0 then + wrong = math.random(1, 10) + else + wrong = utils.clamp(wrong, 0, wrong) + end + return {right, wrong} + end, + + -- 3 : The sum total of each one of your current elemental resistance levels is? + [3] = function(player) + local element_total = 0 + for i = xi.mod.FIRE_RES_RANK, xi.mod.DARK_RES_RANK do + element_total = element_total + player:getMod(i) + end + local right = element_total + local wrong = right + (randomSign() * math.random(1, 10)) + return {right, wrong} + end, + + -- 4 : The total number of foes you’ve felled is? + + -- 5 : Multiply your current attack and defense, and what do you get!? + [5] = function(player) + local right = player:getStat(xi.mod.ATT) * player:getStat(xi.mod.DEF) + local wrong = right + (randomSign() * math.random(21, 88)) + return {right, wrong} + end, + + -- 6 : The total number of times you’ve strolled through the doors of your Mog House is? + -- 7 : The total number of times you’ve been incapacitated by your enemies is? + -- 8 : The total number of times you’ve participated in a party is? + -- 9 : The total number of times you’ve affiliated yourself with an alliance is? +} + +local assignRandomTriviaQuestions = function(player) + local questions = {} + for i, _ in pairs(xi.amk.helpers.triviaQuestions) do + table.insert(questions, i) + end + + for i = 1, 3 do + local index = math.random(1, #questions) + player:setLocalVar('Mission[10][12][p2]question' .. i, questions[index]) + table.remove(questions, index) + end +end + +local resetPuzzleVars = function(player) + player:setLocalVar('Mission[10][12][p2]progress', 0) + player:setLocalVar('Mission[10][12][p2]timeLimit', 0) + player:setLocalVar('Mission[10][12][p2]question1', 0) + player:setLocalVar('Mission[10][12][p2]question2', 0) + player:setLocalVar('Mission[10][12][p2]question3', 0) + player:setLocalVar('Mission[10][12][p2]correctStooge', 0) + player:setLocalVar('Mission[10][12][p2]previousStooge', 0) +end + +local stooges = +{ + [XARC.npc.OPTION_ONE] = { option = 0, stoogeNum = 1, answerOne = 1, answerTwo = 2}, + [XARC.npc.OPTION_TWO] = { option = 1, stoogeNum = 2, answerOne = 0, answerTwo = 2}, + [XARC.npc.OPTION_THREE] = { option = 2, stoogeNum = 3, answerOne = 0, answerTwo = 1}, +} + +xi.amk.helpers.puzzleTwoOnTrigger = function(player, npc, mission) + local p2Progress = player:getLocalVar('Mission[10][12][p2]progress') + if p2Progress == 0 then + p2Progress = 1 + player:setLocalVar('Mission[10][12][p2]progress', p2Progress) + assignRandomTriviaQuestions(player) + end + + local currentQuestion = player:getLocalVar('Mission[10][12][p2]question' .. p2Progress) + local correctStooge = player:getLocalVar('Mission[10][12][p2]correctStooge') + local previousStooge = player:getLocalVar('Mission[10][12][p2]previousStooge') + local stoogeNum = stooges[npc:getID()].stoogeNum + local timeLimit = player:getLocalVar('Mission[10][12][p2]timeLimit') + + -- DEBUG + -- player:PrintToPlayer(string.format("p2Progress: %s, time_limit: %s, currentQuestion: %s, correctStooge: %s, previousStooge: %s, stoogeNum: %s", + -- p2Progress, timeLimit, currentQuestion, correctStooge, previousStooge, stoogeNum)) + + if previousStooge ~= stoogeNum then + return mission:progressEvent( + 200, + p2Progress, + currentQuestion, + correctStooge, + previousStooge, + stoogeNum, + timeLimit, + os.time() + ) + end + +end + +xi.amk.helpers.puzzleTwoOnEventUpdate = function(player, csid, option, npc, mission) + local stooge = stooges[npc:getID()] + + if option >= 1 and option <= 3 then + local timeLimit = os.time() + 180 -- Three minutes + local currentQuestion = player:getLocalVar('Mission[10][12][p2]question' .. player:getLocalVar('Mission[10][12][p2]progress')) + local answers = xi.amk.helpers.triviaQuestions[currentQuestion](player) + + -- Randomize which of the two other options is correct. Set vars to default, then swap only when answerOne is 1 + local answerOne = math.random(1,2) + local answerTwo = 1 + local correctOption = stooge.answerTwo + + if answerOne == 1 then + -- Swap variables + answerTwo = 2 + correctOption = stooge.answerOne + end + + player:setLocalVar('Mission[10][12][p2]correctStooge', correctOption + 1) -- Stooge num is always 1 more than the option + player:setLocalVar('Mission[10][12][p2]timeLimit', timeLimit) + player:setLocalVar('Mission[10][12][p2]previousStooge', stooge.option + 1) + + player:updateEvent( + answers[answerOne], + stooge.answerOne, + answers[answerTwo], + stooge.answerTwo, + currentQuestion, + timeLimit, + os.time() + ) + elseif option == 11 then + -- Incorrect Answer, reset all vars (no need to zone) + resetPuzzleVars(player) + local randomFlavorText = math.random(1,2) + if randomFlavorText == 1 then + player:messageSpecial(XARC.text.INCORRECT_NO_GIFTS) + else + player:messageSpecial(XARC.text.EXACTLY_WRONG) + end + end +end + +xi.amk.helpers.puzzleTwoOnEventFinish = function(player, csid, option, npc, mission) + local p2Progress = player:getLocalVar('Mission[10][12][p2]progress') + if csid == 200 then + if option == 1 then + player:setLocalVar('Mission[10][12][p2]progress', p2Progress + 1) + elseif option == 2 and p2Progress == 4 then + -- Won game, reset all vars + resetPuzzleVars(player) + npcUtil.giveKeyItem(player, xi.ki.GAUNTLET_CHALLENGE_KUPON) + player:setLocalVar('Mission[10][12][p2]progress', 10) + + -- Advance to puzzle 3 + -- player:setCharVar('Mission[10][12]progress', 3) + -- player:setCharVar('Mission[10][12][p3]stoogeArg', 1) + end + end +end \ No newline at end of file diff --git a/scripts/zones/Xarcabard/DefaultActions.lua b/scripts/zones/Xarcabard/DefaultActions.lua index dd57cbb2d02..f6aaedfbd53 100644 --- a/scripts/zones/Xarcabard/DefaultActions.lua +++ b/scripts/zones/Xarcabard/DefaultActions.lua @@ -1,8 +1,11 @@ local ID = zones[xi.zone.XARCABARD] return { - ['qm2'] = { messageSpecial = ID.text.ONLY_SHARDS }, - ['qm3'] = { messageSpecial = ID.text.ONLY_SHARDS }, - ['qm4'] = { messageSpecial = ID.text.ONLY_SHARDS }, - ['qm5'] = { messageSpecial = ID.text.NOTHING_OUT_OF_ORDINARY }, + ['Option_One'] = { messageSpecial = ID.text.NOTHING_OUT_OF_ORDINARY }, + ['Option_Two'] = { messageSpecial = ID.text.NOTHING_OUT_OF_ORDINARY }, + ['Option_Three'] = { messageSpecial = ID.text.NOTHING_OUT_OF_ORDINARY }, + ['qm2'] = { messageSpecial = ID.text.ONLY_SHARDS }, + ['qm3'] = { messageSpecial = ID.text.ONLY_SHARDS }, + ['qm4'] = { messageSpecial = ID.text.ONLY_SHARDS }, + ['qm5'] = { messageSpecial = ID.text.NOTHING_OUT_OF_ORDINARY }, } diff --git a/scripts/zones/Xarcabard/IDs.lua b/scripts/zones/Xarcabard/IDs.lua index 9256149fecd..f234673883a 100644 --- a/scripts/zones/Xarcabard/IDs.lua +++ b/scripts/zones/Xarcabard/IDs.lua @@ -31,6 +31,8 @@ zones[xi.zone.XARCABARD] = PLAYERS_HAVE_NOT_REACHED_LEVEL = 7855, -- Players who have not reached level are prohibited from entering Dynamis. DYNA_NPC_DEFAULT_MESSAGE = 7892, -- There is a symbol carved into the rock here. GARRISON_BASE = 8047, -- Hm? What is this? %? How do I know this is not some [San d'Orian/Bastokan/Windurstian] trick? + INCORRECT_NO_GIFTS = 8153, -- Ohhhhhh, I'm sorry, that's incorrect. But don't worry, we have some lovely parting gifts for-- What's that? We don't? + EXACTLY_WRONG = 8154, -- Why that's exactly rrr-wrong! I'm soooooo sorry, but it looks like you'll have to start all over again! But you'll get 'em next time. Right, kupo? PLAYER_OBTAINS_ITEM = 8170, -- obtains ! UNABLE_TO_OBTAIN_ITEM = 8171, -- You were unable to obtain the item. PLAYER_OBTAINS_TEMP_ITEM = 8172, -- obtains the temporary item: ! @@ -112,10 +114,13 @@ zones[xi.zone.XARCABARD] = }, npc = { - OVERSEER_BASE = GetFirstID('Jeantelas_RK'), - BOREAL_TIGER_QM = GetFirstID('qm_boreal_tiger'), - BOREAL_COEURL_QM = GetFirstID('qm_boreal_coeurl'), - BOREAL_HOUND_QM = GetFirstID('qm_boreal_hound'), + OVERSEER_BASE = GetFirstID('Jeantelas_RK'), + BOREAL_TIGER_QM = GetFirstID('qm_boreal_tiger'), + BOREAL_COEURL_QM = GetFirstID('qm_boreal_coeurl'), + BOREAL_HOUND_QM = GetFirstID('qm_boreal_hound'), + OPTION_ONE = GetFirstID('Option_One'), + OPTION_TWO = GetFirstID('Option_Two'), + OPTION_THREE = GetFirstID('Option_Three'), }, } diff --git a/scripts/zones/Xarcabard/npcs/Option_One.lua b/scripts/zones/Xarcabard/npcs/Option_One.lua new file mode 100644 index 00000000000..da54008647c --- /dev/null +++ b/scripts/zones/Xarcabard/npcs/Option_One.lua @@ -0,0 +1,22 @@ +----------------------------------- +-- Area: Xarcabard +-- NPC: Option One +-- Involved in Mission: A Moogle Kupo d'Etat mission 13 - A Challenge! You Could Be a Winner +-- !pos 126 -24 -118 112 +----------------------------------- +local entity = {} + +entity.onTrade = function(player, npc, trade) +end + +entity.onTrigger = function(player, npc) + xi.amk.helpers.assignRandomTriviaQuestions(player) +end + +entity.onEventUpdate = function(player, csid, option, npc) +end + +entity.onEventFinish = function(player, csid, option, npc) +end + +return entity diff --git a/scripts/zones/Xarcabard/npcs/Option_Three.lua b/scripts/zones/Xarcabard/npcs/Option_Three.lua new file mode 100644 index 00000000000..ed051e4b3b6 --- /dev/null +++ b/scripts/zones/Xarcabard/npcs/Option_Three.lua @@ -0,0 +1,21 @@ +----------------------------------- +-- Area: Xarcabard +-- NPC: Option Three +-- Involved in Mission: A Moogle Kupo d'Etat mission 13 - A Challenge! You Could Be a Winner +-- !pos 1 -23 -103 112 +----------------------------------- +local entity = {} + +entity.onTrade = function(player, npc, trade) +end + +entity.onTrigger = function(player, npc) +end + +entity.onEventUpdate = function(player, csid, option, npc) +end + +entity.onEventFinish = function(player, csid, option, npc) +end + +return entity diff --git a/scripts/zones/Xarcabard/npcs/Option_Two.lua b/scripts/zones/Xarcabard/npcs/Option_Two.lua new file mode 100644 index 00000000000..022481dd417 --- /dev/null +++ b/scripts/zones/Xarcabard/npcs/Option_Two.lua @@ -0,0 +1,21 @@ +----------------------------------- +-- Area: Xarcabard +-- NPC: Option Two +-- Involved in Mission: A Moogle Kupo d'Etat mission 13 - A Challenge! You Could Be a Winner +-- !pos 66 -24 -191 112 +----------------------------------- +local entity = {} + +entity.onTrade = function(player, npc, trade) +end + +entity.onTrigger = function(player, npc) +end + +entity.onEventUpdate = function(player, csid, option, npc) +end + +entity.onEventFinish = function(player, csid, option, npc) +end + +return entity