Merge pull request #771 from argonui/clue-replenishing

Added clue replenishing to hotkey
This commit is contained in:
dscarpac 2024-07-19 13:09:27 -07:00 committed by GitHub
commit ba8f6c9c3e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 153 additions and 66 deletions

View File

@ -276,6 +276,12 @@ function removeOneUse(playerColor, hoveredObject)
end
end
-- error handling
if not targetObject then
broadcastToColor("No tokens found!", playerColor, "Yellow")
return
end
-- release sealed token if card has one and no uses
if tokenChecker.isChaosToken(targetObject) and hoveredObject.hasTag("CardThatSeals") then
local func = hoveredObject.getVar("releaseOneToken") -- check if function exists
@ -285,12 +291,6 @@ function removeOneUse(playerColor, hoveredObject)
end
end
-- error handling
if not targetObject then
broadcastToColor("No tokens found!", playerColor, "Yellow")
return
end
-- handling for stacked tokens
if targetObject.getQuantity() > 1 then
targetObject = targetObject.takeObject()

View File

@ -464,7 +464,7 @@ end
-- Count victory points from locations in play area
---@param highlightOff boolean True if highlighting should be enabled
---@return. Returns the total amount of VP found in the play area
---@return number totalVP Total amount of VP found in the play area
function countVP(highlightOff)
local totalVP = 0

View File

@ -219,9 +219,6 @@ do
end
for i = 1, tokenCount do
offsets[i] = card.positionToWorld(PLAYER_CARD_TOKEN_OFFSETS[tokenCount][i])
-- Fix the y-position for the spawn, since positionToWorld considers rotation which can
-- have bad results for face up/down differences
offsets[i].y = card.getPosition().y + 0.15
end
end
@ -250,7 +247,6 @@ do
elseif tokenType == "universalActionAbility" then
local matColor = playermatApi.getMatColorByPosition(card.getPosition())
local class = playermatApi.returnInvestigatorClass(matColor)
callback = function(spawned) spawned.call("updateClassAndSymbol", { class = class, symbol = subType or class }) end
end
@ -299,9 +295,10 @@ do
---@param card tts__Object Card object to be replenished
---@param uses table The already decoded metadata.uses (to avoid decoding again)
TokenManager.maybeReplenishCard = function(card, uses)
-- TODO: support for cards with multiple uses AND replenish (as of yet, no official card needs that)
if uses[1].count and uses[1].replenish then
internal.replenishTokens(card, uses)
for _, useInfo in ipairs(uses) do
if useInfo.count and useInfo.replenish then
internal.replenishTokens(card, useInfo)
end
end
end
@ -459,59 +456,98 @@ do
return nil
end
-- Dynamically create positions for clues on a card.
-- Dynamically create positions for clues on a card
---@param card tts__Object Card the clues will be placed on
---@param count number How many clues?
---@return table: Array of global positions to spawn the clues at
internal.buildClueOffsets = function(card, count)
-- make sure clues always spawn from left to right
local modifier = card.is_face_down and 1 or -1
local cluePositions = {}
for i = 1, count do
local row = math.floor(1 + (i - 1) / 4)
local column = (i - 1) % 4
local cluePos = card.positionToWorld(Vector(-0.825 + 0.55 * column, 0, -1.5 + 0.55 * row))
cluePos.y = cluePos.y + 0.05
-- get the set number (1 for clue 1-16, 2 for 17-32 etc.)
local set = math.floor((i - 1) / 16) + 1
-- get the local index (always number from 1-16)
local localIndex = (i - 1) % 16
-- get row and column for this clue
local row = math.floor(localIndex / 4) + 1
local column = localIndex % 4
-- calculate local position
local localPos = Vector((-0.825 + 0.55 * column) * modifier, 0, -1.5 + 0.55 * row)
-- get the global clue position (higher y-position for each set)
local cluePos = card.positionToWorld(localPos) + Vector(0, 0.03 + 0.103 * (set - 1), 0)
-- add position to table
table.insert(cluePositions, cluePos)
end
return cluePositions
end
---@param card tts__Object Card object to be replenished
---@param uses table The already decoded metadata.uses (to avoid decoding again)
internal.replenishTokens = function(card, uses)
---@param useInfo table The already decoded subtable of metadata.uses (to avoid decoding again)
internal.replenishTokens = function(card, useInfo)
-- get current amount of matching resource tokens on the card
local clickableResourceCounter = nil
local foundTokens = 0
local searchType = string.lower(uses[1].type)
for _, obj in ipairs(searchLib.onObject(card, "isTileOrToken")) do
local memo = obj.getMemo()
if searchType == memo then
local maybeDeleteThese = {}
if useInfo.token == "clue" then
for _, obj in ipairs(searchLib.onObject(card, "isClue")) do
foundTokens = foundTokens + math.abs(obj.getQuantity())
obj.destruct()
elseif memo == "resourceCounter" then
foundTokens = obj.getVar("val")
clickableResourceCounter = obj
break
table.insert(maybeDeleteThese, obj)
end
elseif useInfo.token == "doom" then
for _, obj in ipairs(searchLib.onObject(card, "isDoom")) do
foundTokens = foundTokens + math.abs(obj.getQuantity())
table.insert(maybeDeleteThese, obj)
end
else
-- search for the token instead if there's no special resource state for it
local searchType = string.lower(useInfo.type)
if stateTable[searchType] == nil then
searchType = useInfo.token
end
for _, obj in ipairs(searchLib.onObject(card, "isTileOrToken")) do
local memo = obj.getMemo()
if searchType == memo then
foundTokens = foundTokens + math.abs(obj.getQuantity())
table.insert(maybeDeleteThese, obj)
elseif memo == "resourceCounter" then
foundTokens = obj.getVar("val")
clickableResourceCounter = obj
break
end
end
end
-- this is the theoretical new amount of uses (to be checked below)
local newCount = foundTokens + uses[1].replenish
local newCount = foundTokens + useInfo.replenish
-- if there are already more uses than the replenish amount, keep them
if foundTokens > uses[1].count then
if foundTokens > useInfo.count then
newCount = foundTokens
-- only replenish up until the replenish amount
elseif newCount > uses[1].count then
newCount = uses[1].count
elseif newCount > useInfo.count then
newCount = useInfo.count
end
-- update the clickable counter or spawn a group of tokens
if clickableResourceCounter then
clickableResourceCounter.call("updateVal", newCount)
else
TokenManager.spawnTokenGroup(card, uses[1].token, newCount, _, uses[1].type)
-- delete existing tokens
for _, obj in ipairs(maybeDeleteThese) do
obj.destruct()
end
-- spawn new token group
TokenManager.spawnTokenGroup(card, useInfo.token, newCount, _, useInfo.type)
end
end

View File

@ -5,6 +5,7 @@ do
isDeck = function(x) return x.type == "Deck" end,
isCardOrDeck = function(x) return x.type == "Card" or x.type == "Deck" end,
isClue = function(x) return x.memo == "clueDoom" and x.is_face_down == false end,
isDoom = function(x) return x.memo == "clueDoom" and x.is_face_down == true end,
isTileOrToken = function(x) return x.type == "Tile" end,
isUniversalToken = function(x) return x.getMemo() == "universalActionAbility" end,
}

View File

@ -1,4 +1,3 @@
local guidReferenceApi = require("core/GUIDReferenceApi")
local playermatApi = require("playermat/PlayermatApi")
local searchLib = require("util/SearchLib")
local tokenManager = require("core/token/TokenManager")
@ -24,35 +23,23 @@ function onScriptingButtonDown(index, playerColor)
-- check for subtype of resource based on card below
if tokenType == "resource" then
local card
local hoverObj = Player[playerColor].getHoverObject()
if hoverObj and hoverObj.type == "Card" then
card = hoverObj
elseif hoverObj then
-- use the first card below the hovered object if it's not a card1
for _, obj in ipairs(searchLib.belowPosition(position, "isCard")) do
card = obj
break
end
local card = getTargetCard(playerColor, position)
if card and not card.is_face_down then
local status = addUseToCard(card, tokenType)
if status == true then return end
end
-- get the metadata from the card and maybe replenish a use
if card and not card.is_face_down then
local metadata = JSON.decode(card.getGMNotes()) or {}
local uses = metadata.uses or {}
for _, useInfo in ipairs(uses) do
if useInfo.token == "resource" then
-- artifically create replenish data to re-use that existing functionality
uses[1].count = 999
uses[1].replenish = 1
local matColor = playermatApi.getMatColorByPosition(position)
local mat = guidReferenceApi.getObjectByOwnerAndType(matColor, "Playermat")
tokenManager.maybeReplenishCard(card, uses, mat)
return
end
end
-- check hovered object for location data or 'uses (x clues)' and add one
elseif tokenType == "clue" then
local card = getTargetCard(playerColor, position)
if card and (not card.is_face_down or card.hasTag("Location")) then
local status = addUseToCard(card, tokenType)
if status == true then return end
end
-- check hovered object for "resourceCounter" tokens and increase them instead
-- check hovered object for "resourceCounter" tokens and increase them instead
elseif tokenType == "resourceCounter" then
local hoverObj = Player[playerColor].getHoverObject()
if hoverObj then
@ -61,7 +48,8 @@ function onScriptingButtonDown(index, playerColor)
return
end
end
-- check hovered object for "damage" and "horror" tokens and increase them instead
-- check hovered object for "damage" and "horror" tokens and increase them instead
elseif tokenType == "damage" or tokenType == "horror" then
local hoverObj = Player[playerColor].getHoverObject()
if hoverObj then
@ -74,12 +62,74 @@ function onScriptingButtonDown(index, playerColor)
end
end
end
-- check for nearest investigator card and change action token state to its class
-- check for nearest investigator card and change action token state to its class
elseif tokenType == "universalActionAbility" then
local matColor = playermatApi.getMatColorByPosition(position)
local matRotation = playermatApi.returnRotation(matColor)
local class = playermatApi.returnInvestigatorClass(matColor)
callback = function(spawned) spawned.call("updateClassAndSymbol", { class = class, symbol = class }) end
callback = function(spawned)
spawned.setRotation(matRotation)
spawned.call("updateClassAndSymbol", { class = class, symbol = class })
end
end
tokenManager.spawnToken(position, tokenType, rotation, callback)
end
-- gets the target card for this operation
---@param playerColor string Color of the triggering player
---@param position tts__Vector Position to check for a card (if there isn't a hovered card)
function getTargetCard(playerColor, position)
local hoverObj = Player[playerColor].getHoverObject()
if hoverObj and hoverObj.type == "Card" then
return hoverObj
elseif hoverObj then
-- use the first card below the hovered object if it's not a card
for _, obj in ipairs(searchLib.belowPosition(position, "isCard")) do
return obj
end
end
end
-- adds a use to a card (TODO: probably move this to the TokenManager?)
---@param card tts__Object Card that should get a use added
---@param useType string Type of uses to be added
function addUseToCard(card, useType)
local metadata = JSON.decode(card.getGMNotes()) or {}
-- get correct data for location
if metadata.type == "Location" then
if not card.is_face_down and metadata.locationFront ~= nil then
metadata = metadata.locationFront
elseif metadata.locationBack ~= nil then
metadata = metadata.locationBack
end
-- if there are no uses at all, add "empty" uses for fake replenishing (only for clues)
if metadata.uses == nil then
metadata.uses = { { token = "clue" } }
end
end
local match = false
for _, useInfo in ipairs(metadata.uses) do
if useInfo.token == useType then
-- artificially create replenish data to re-use that existing functionality
useInfo.count = 999
useInfo.replenish = 1
match = true
else
-- artificially disable other uses from replenishing
useInfo.replenish = nil
end
end
-- if matching uses were found, perform the "fake" replenish
if match then
tokenManager.maybeReplenishCard(card, metadata.uses)
return true
else
return false
end
end