Vs skill (#44)

* vs skill concept

* unify card:clone

* virtual card

* wusheng

* exppattern

* use virtual card(WIP)

* change cardId to card

* virtual card log

* notify skill invoked

* fix coroutine bug; allow vsskill to response

* extra_data for askForUseCard

Co-authored-by: Ho-spair <linyuy@163.com>
This commit is contained in:
notify 2023-01-16 19:13:07 +08:00 committed by GitHub
parent 88d40018db
commit b6530eae9d
18 changed files with 857 additions and 280 deletions

View File

@ -160,7 +160,7 @@ end
function GetSkillData(skill_name) function GetSkillData(skill_name)
local skill = Fk.skills[skill_name] local skill = Fk.skills[skill_name]
local freq = "notactive" local freq = "notactive"
if skill:isInstanceOf(ActiveSkill) then if skill:isInstanceOf(ActiveSkill) or skill:isInstanceOf(ViewAsSkill) then
freq = "active" freq = "active"
end end
return json.encode{ return json.encode{
@ -173,8 +173,12 @@ end
function ActiveCanUse(skill_name) function ActiveCanUse(skill_name)
local skill = Fk.skills[skill_name] local skill = Fk.skills[skill_name]
local ret = false local ret = false
if skill and skill:isInstanceOf(ActiveSkill) then if skill then
ret = skill:canUse(Self) if skill:isInstanceOf(ActiveSkill) then
ret = skill:canUse(Self)
elseif skill:isInstanceOf(ViewAsSkill) then
ret = skill:enabledAtPlay(Self)
end
end end
return json.encode(ret) return json.encode(ret)
end end
@ -182,8 +186,12 @@ end
function ActiveCardFilter(skill_name, to_select, selected, selected_targets) function ActiveCardFilter(skill_name, to_select, selected, selected_targets)
local skill = Fk.skills[skill_name] local skill = Fk.skills[skill_name]
local ret = false local ret = false
if skill and skill:isInstanceOf(ActiveSkill) then if skill then
ret = skill:cardFilter(to_select, selected, selected_targets) if skill:isInstanceOf(ActiveSkill) then
ret = skill:cardFilter(to_select, selected, selected_targets)
elseif skill:isInstanceOf(ViewAsSkill) then
ret = skill:cardFilter(to_select, selected)
end
end end
return json.encode(ret) return json.encode(ret)
end end
@ -191,8 +199,15 @@ end
function ActiveTargetFilter(skill_name, to_select, selected, selected_cards) function ActiveTargetFilter(skill_name, to_select, selected, selected_cards)
local skill = Fk.skills[skill_name] local skill = Fk.skills[skill_name]
local ret = false local ret = false
if skill and skill:isInstanceOf(ActiveSkill) then if skill then
ret = skill:targetFilter(to_select, selected, selected_cards) if skill:isInstanceOf(ActiveSkill) then
ret = skill:targetFilter(to_select, selected, selected_cards)
elseif skill:isInstanceOf(ViewAsSkill) then
local card = skill:viewAs(selected_cards)
if card then
ret = card.skill:targetFilter(to_select, selected, selected_cards)
end
end
end end
return json.encode(ret) return json.encode(ret)
end end
@ -200,15 +215,46 @@ end
function ActiveFeasible(skill_name, selected, selected_cards) function ActiveFeasible(skill_name, selected, selected_cards)
local skill = Fk.skills[skill_name] local skill = Fk.skills[skill_name]
local ret = false local ret = false
if skill and skill:isInstanceOf(ActiveSkill) then if skill then
ret = skill:feasible(selected, selected_cards) if skill:isInstanceOf(ActiveSkill) then
ret = skill:feasible(selected, selected_cards)
elseif skill:isInstanceOf(ViewAsSkill) then
local card = skill:viewAs(selected_cards)
if card then
ret = card.skill:feasible(selected, selected_cards)
end
end
end end
return json.encode(ret) return json.encode(ret)
end end
-- ViewAsSkill (Todo)
function CanViewAs(skill_name, card_ids) function CanViewAs(skill_name, card_ids)
return "true" local skill = Fk.skills[skill_name]
local ret = false
if skill then
if skill:isInstanceOf(ViewAsSkill) then
ret = skill:viewAs(card_ids) ~= nil
elseif skill:isInstanceOf(ActiveSkill) then
ret = true
end
end
return json.encode(ret)
end
function CardFitPattern(card_name, pattern)
local exp = Exppattern:Parse(pattern)
local ret = exp:matchExp(card_name)
return json.encode(ret)
end
function SkillFitPattern(skill_name, pattern)
local skill = Fk.skills[skill_name]
local ret = false
if skill and skill.pattern then
local exp = Exppattern:Parse(pattern)
ret = exp:matchExp(skill.pattern)
end
return json.encode(ret)
end end
Fk:loadTranslationTable{ Fk:loadTranslationTable{
@ -355,10 +401,22 @@ Fk:loadTranslationTable{
-- useCard -- useCard
["#UseCard"] = "%from 使用了牌 %card", ["#UseCard"] = "%from 使用了牌 %card",
["#UseCardToTargets"] = "%from 使用了牌 %card目标是 %to", ["#UseCardToTargets"] = "%from 使用了牌 %card目标是 %to",
["#CardUseCollaborator"] = "%from 在此次 %card 中的子目标是 %to", ["#CardUseCollaborator"] = "%from 在此次 %arg 中的子目标是 %to",
["#UseCardToCard"] = "%from 使用了牌 %card目标是 %arg", ["#UseCardToCard"] = "%from 使用了牌 %card目标是 %arg",
["#ResponsePlayCard"] = "%from 打出了牌 %card", ["#ResponsePlayCard"] = "%from 打出了牌 %card",
["#UseVCard"] = "%from 将 %card 当 %arg 使用",
["#UseVCardToTargets"] = "%from 将 %card 当 %arg 使用,目标是 %to",
["#UseVCardToCard"] = "%from 将 %card 当 %arg2 使用,目标是 %arg",
["#ResponsePlayVCard"] = "%from 将 %card 当 %arg 打出",
["#UseV0Card"] = "%from 使用了 %arg",
["#UseV0CardToTargets"] = "%from 使用了 %arg目标是 %to",
["#UseV0CardToCard"] = "%from 使用了 %arg2目标是 %arg",
["#ResponsePlayV0Card"] = "%from 打出了 %arg",
-- skill
["#InvokeSkill"] = "%from 发动了 “%arg”",
-- judge -- judge
["#StartJudgeReason"] = "%from 开始了 %arg 的判定", ["#StartJudgeReason"] = "%from 开始了 %arg 的判定",
["#InitialJudge"] = "%from 的判定牌为 %card", ["#InitialJudge"] = "%from 的判定牌为 %card",

View File

@ -9,6 +9,7 @@
---@field type CardType ---@field type CardType
---@field sub_type CardSubtype ---@field sub_type CardSubtype
---@field area CardArea ---@field area CardArea
---@field subcards integer[]
local Card = class("Card") local Card = class("Card")
---@alias Suit integer ---@alias Suit integer
@ -74,6 +75,74 @@ function Card:initialize(name, suit, number, color)
self.type = 0 self.type = 0
self.sub_type = Card.SubTypeNone self.sub_type = Card.SubTypeNone
self.skill = nil self.skill = nil
self.subcards = {}
end
---@param suit Suit
---@param number integer
---@return Card
function Card:clone(suit, number)
local newCard = self.class:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
function Card:isVirtual()
return self.id <= 0
end
function Card:getEffectiveId()
if self:isVirtual() then
return #self.subcards > 0 and self.subcards[1] or nil
end
return self.id
end
local function updateColorAndNumber(card)
local color = Card.NoColor
local number = 0
local different_color = false
for _, id in ipairs(card.subcards) do
local c = Fk:getCardById(id)
number = math.min(number + c.number, 13)
if color ~= c.color then
if not different_color then
if color ~= Card.NoColor then
different_color = true
end
color = c.color
else
color = Card.NoColor
end
end
end
card.color = color
card.number = number
end
---@param card integer|Card
function Card:addSubcard(card)
if type(card) == "number" then
table.insert(self.subcards, card)
else
assert(card:isInstanceOf(Card))
assert(not card:isVirtual(), "Can not add virtual card as subcard")
table.insert(self.subcards, card.id)
end
updateColorAndNumber(self)
end
function Card:addSubcards(cards)
for _, c in ipairs(cards) do
self:addSubcard(c)
end
end
function Card:clearSubcards()
self.subcards = {}
updateColorAndNumber(self)
end end
function Card:getSuitString() function Card:getSuitString()
@ -91,6 +160,16 @@ function Card:getSuitString()
end end
end end
function Card:getColorString()
local color = self.color
if color == Card.Black then
return "black"
elseif color == Card.Red then
return "red"
end
return "nocolor"
end
local function getNumberStr(num) local function getNumberStr(num)
if num == 1 then if num == 1 then
return "A" return "A"
@ -107,9 +186,13 @@ end
-- for sendLog -- for sendLog
function Card:toLogString() function Card:toLogString()
local ret = string.format('<font color="#0598BC"><b>%s</b></font>', Fk:translate(self.name) .. "[") local ret = string.format('<font color="#0598BC"><b>%s</b></font>', Fk:translate(self.name) .. "[")
ret = ret .. Fk:translate("log_" .. self:getSuitString()) if self:isVirtual() then
if self.number > 0 then ret = ret .. Fk:translate(self:getColorString())
ret = ret .. string.format('<font color="%s"><b>%s</b></font>', self.color == Card.Red and "#CC3131" or "black", getNumberStr(self.number)) else
ret = ret .. Fk:translate("log_" .. self:getSuitString())
if self.number > 0 then
ret = ret .. string.format('<font color="%s"><b>%s</b></font>', self.color == Card.Red and "#CC3131" or "black", getNumberStr(self.number))
end
end end
ret = ret .. '<font color="#0598BC"><b>]</b></font>' ret = ret .. '<font color="#0598BC"><b>]</b></font>'
return ret return ret

View File

@ -6,13 +6,4 @@ function BasicCard:initialize(name, suit, number)
self.type = Card.TypeBasic self.type = Card.TypeBasic
end end
---@param suit Suit
---@param number integer
---@return BasicCard
function BasicCard:clone(suit, number)
local newCard = BasicCard:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
return BasicCard return BasicCard

View File

@ -17,15 +17,6 @@ function Weapon:initialize(name, suit, number, attackRange)
self.attack_range = attackRange or 1 self.attack_range = attackRange or 1
end end
---@param suit Suit
---@param number integer
---@return Weapon
function Weapon:clone(suit, number)
local newCard = Weapon:new(self.name, suit, number, self.attack_range)
newCard.skill = self.skill
return newCard
end
---@class Armor : EquipCard ---@class Armor : EquipCard
local Armor = EquipCard:subclass("armor") local Armor = EquipCard:subclass("armor")
@ -34,15 +25,6 @@ function Armor:initialize(name, suit, number)
self.sub_type = Card.SubtypeArmor self.sub_type = Card.SubtypeArmor
end end
---@param suit Suit
---@param number integer
---@return Armor
function Armor:clone(suit, number)
local newCard = Armor:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
---@class DefensiveRide : EquipCard ---@class DefensiveRide : EquipCard
local DefensiveRide = EquipCard:subclass("DefensiveRide") local DefensiveRide = EquipCard:subclass("DefensiveRide")
@ -51,15 +33,6 @@ function DefensiveRide:initialize(name, suit, number)
self.sub_type = Card.SubtypeDefensiveRide self.sub_type = Card.SubtypeDefensiveRide
end end
---@param suit Suit
---@param number integer
---@return DefensiveRide
function DefensiveRide:clone(suit, number)
local newCard = DefensiveRide:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
---@class OffensiveRide : EquipCard ---@class OffensiveRide : EquipCard
local OffensiveRide = EquipCard:subclass("OffensiveRide") local OffensiveRide = EquipCard:subclass("OffensiveRide")
@ -68,15 +41,6 @@ function OffensiveRide:initialize(name, suit, number)
self.sub_type = Card.SubtypeOffensiveRide self.sub_type = Card.SubtypeOffensiveRide
end end
---@param suit Suit
---@param number integer
---@return OffensiveRide
function OffensiveRide:clone(suit, number)
local newCard = OffensiveRide:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
---@class Treasure : EquipCard ---@class Treasure : EquipCard
local Treasure = EquipCard:subclass("Treasure") local Treasure = EquipCard:subclass("Treasure")
@ -85,13 +49,4 @@ function Treasure:initialize(name, suit, number)
self.sub_type = Card.SubtypeTreasure self.sub_type = Card.SubtypeTreasure
end end
---@param suit Suit
---@param number integer
---@return Treasure
function Treasure:clone(suit, number)
local newCard = Treasure:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
return { EquipCard, Weapon, Armor, DefensiveRide, OffensiveRide, Treasure } return { EquipCard, Weapon, Armor, DefensiveRide, OffensiveRide, Treasure }

View File

@ -6,17 +6,6 @@ function TrickCard:initialize(name, suit, number)
self.type = Card.TypeTrick self.type = Card.TypeTrick
end end
---@param suit Suit
---@param number integer
---@return TrickCard
function TrickCard:clone(suit, number)
local newCard = TrickCard:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
---@class DelayedTrickCard : TrickCard ---@class DelayedTrickCard : TrickCard
local DelayedTrickCard = TrickCard:subclass("DelayedTrickCard") local DelayedTrickCard = TrickCard:subclass("DelayedTrickCard")
@ -25,13 +14,4 @@ function DelayedTrickCard:initialize(name, suit, number)
self.sub_type = Card.SubtypeDelayedTrick self.sub_type = Card.SubtypeDelayedTrick
end end
---@param suit Suit
---@param number integer
---@return DelayedTrickCard
function DelayedTrickCard:clone(suit, number)
local newCard = DelayedTrickCard:new(self.name, suit, number)
newCard.skill = self.skill
return newCard
end
return { TrickCard, DelayedTrickCard } return { TrickCard, DelayedTrickCard }

View File

@ -137,12 +137,14 @@ function Engine:addGenerals(generals)
end end
local cardId = 1 local cardId = 1
local _card_name_table = {}
---@param card Card ---@param card Card
function Engine:addCard(card) function Engine:addCard(card)
assert(card.class:isSubclassOf(Card)) assert(card.class:isSubclassOf(Card))
card.id = cardId card.id = cardId
cardId = cardId + 1 cardId = cardId + 1
table.insert(self.cards, card) table.insert(self.cards, card)
_card_name_table[card.name] = card
end end
---@param cards Card[] ---@param cards Card[]
@ -152,6 +154,16 @@ function Engine:addCards(cards)
end end
end end
---@param name string
---@param suit Suit
---@param number integer
---@return Card
function Engine:cloneCard(name, suit, number)
local cd = _card_name_table[name]
assert(cd, "Attempt to clone a card that not added to engine")
return cd:clone(suit, number)
end
---@param num integer ---@param num integer
---@param generalPool General[] ---@param generalPool General[]
---@param except string[] ---@param except string[]

246
lua/core/exppattern.lua Normal file
View File

@ -0,0 +1,246 @@
--[[
Exppattern is a string that describes cards of a same 'type', e.g. name,
suit, etc.
The string will be parsed and construct a new Exppattern instance.
Then we can use this instance to check the card.
Syntax for the string form:
1. the whole string can be splited by ';'. Every slice stands for a Matcher
2. For the matcher string, it can be splited by '|'.
3. And the arrays in class Match is concated by ',' in string.
Example:
slash,jink|2~4|spade;.|.|.|.|.|trick
]]--
---@class Matcher
---@field name string[]
---@field number integer[]
---@field suit integer[]
---@field place string[]
---@field generalName string[]
---@field cardType integer[]
---@field id integer[]
local numbertable = {
["A"] = 1,
["J"] = 11,
["Q"] = 12,
["K"] = 13,
}
local suittable = {
[Card.Spade] = "spade",
[Card.Club] = "club",
[Card.Heart] = "heart",
[Card.Diamond] = "diamond",
}
local typetable = {
[Card.TypeBasic] = "basic",
[Card.TypeTrick] = "trick",
[Card.TypeEquip] = "equip",
}
---@param matcher Matcher
---@param card Card
local function matchCard(matcher, card)
if type(card) == "number" then
card = Fk:getCardById(card)
end
if matcher.name and not table.contains(matcher.name, card.name) then
return false
end
if matcher.number and not table.contains(matcher.number, card.number) then
return false
end
if matcher.suit and not table.contains(matcher.suit, card:getSuitString()) then
return false
end
-- TODO: place
-- TODO: generalName
if matcher.cardType and not table.contains(matcher.cardType, typetable[card.type]) then
return false
end
if matcher.id and not table.contains(matcher.id, card.id) then
return false
end
return true
end
local function hasIntersection(a, b)
if a == nil or b == nil then
return true
end
local tmp = {}
for _, e in ipairs(a) do
tmp[e] = true
end
for _, e in ipairs(b) do
if tmp[e] then
return true
end
end
return false
end
---@param a Matcher
---@param b Matcher
local function matchMatcher(a, b)
local keys = {
"name",
"number",
"suit",
"place",
"generalName",
"cardType",
"id",
}
for _, k in ipairs(keys) do
if not hasIntersection(a[k], b[k]) then
return false
end
end
return true
end
local function parseMatcher(str)
local t = str:split("|")
if #t < 7 then
for i = 1, 7 - #t do
table.insert(t, ".")
end
end
for i, item in ipairs(t) do
t[i] = item:split(",")
end
local ret = {} ---@type Matcher
ret.name = not table.contains(t[1], ".") and t[1] or nil
if not table.contains(t[2], ".") then
ret.number = {}
for _, num in ipairs(t[2]) do
local n = tonumber(num)
if not n then
n = numbertable[num]
end
if n then
table.insertIfNeed(ret.number, n)
else
if string.find(n, "~") then
local start, _end = table.unpack(n:split("~"))
for i = start, _end do
table.insertIfNeed(ret.number, n)
end
end
end
end
end
if not table.contains(t[3], ".") then
ret.suit = {}
for _, num in ipairs(t[3]) do
local n = suittable[num]
if n then
table.insertIfNeed(ret.suit, n)
end
end
end
ret.place = not table.contains(t[4], ".") and t[4] or nil
ret.generalName = not table.contains(t[5], ".") and t[5] or nil
if not table.contains(t[6], ".") then
ret.cardType = {}
for _, num in ipairs(t[6]) do
local n = typetable[num]
if n then
table.insertIfNeed(ret.cardType, n)
end
end
end
if not table.contains(t[7], ".") then
ret.id = {}
for _, num in ipairs(t[6]) do
local n = tonumber(num)
if n and n > 0 then
table.insertIfNeed(ret.id, n)
end
end
end
return ret
end
---@class Exppattern: Object
---@field matchers Matcher[]
local Exppattern = class("Exppattern")
function Exppattern:initialize(spec)
if not spec then
self.matchers = {}
elseif spec[1] ~= nil then
self.matchers = spec
else
self.matchers = {}
self.matchers[1] = spec
end
end
---@param str string
function Exppattern.static:Parse(str)
local ret = Exppattern:new()
local t = str:split(";")
for i, s in ipairs(t) do
ret.matchers[i] = parseMatcher(s)
end
return ret
end
---@param card Card
function Exppattern:match(card)
for _, matcher in ipairs(self.matchers) do
local result = matchCard(matcher, card)
if result then
return true
end
end
return false
end
function Exppattern:matchExp(exp)
if type(exp) == "string" then
exp = Exppattern:Parse(exp)
end
local a = self.matchers
local b = exp.matchers
for _, m in ipairs(a) do
for _, n in ipairs(b) do
if matchMatcher(m, n) then
return true
end
end
end
return false
end
return Exppattern

View File

@ -0,0 +1,35 @@
---@class ViewAsSkill
---@field pattern string @ cards that can be viewAs'ed by this skill
local ViewAsSkill = Skill:subclass("ViewAsSkill")
function ViewAsSkill:initialize(name)
Skill.initialize(self, name, Skill.NotFrequent)
self.pattern = ""
end
---@param to_select integer @ id of a card not selected
---@param selected integer[] @ ids of selected cards
---@return boolean
function ViewAsSkill:cardFilter(to_select, selected)
return false
end
---@param cards integer[] @ ids of cards
---@return card
function ViewAsSkill:viewAs(cards)
return nil
end
-- For extra judgement, like mark or HP
---@param player Player
function ViewAsSkill:enabledAtPlay(player)
return true
end
---@param player Player
function ViewAsSkill:enabledAtResponse(player)
return true
end
return ViewAsSkill

View File

@ -4,6 +4,7 @@ dofile "lua/server/event.lua"
dofile "lua/server/system_enum.lua" dofile "lua/server/system_enum.lua"
TriggerSkill = require "core.skill_type.trigger" TriggerSkill = require "core.skill_type.trigger"
ActiveSkill = require "core.skill_type.active_skill" ActiveSkill = require "core.skill_type.active_skill"
ViewAsSkill = require "core.skill_type.view_as"
DistanceSkill = require "core.skill_type.distance" DistanceSkill = require "core.skill_type.distance"
StatusSkills = { StatusSkills = {
DistanceSkill, DistanceSkill,
@ -111,6 +112,37 @@ function fk.CreateActiveSkill(spec)
return skill return skill
end end
---@class ViewAsSkillSpec: SkillSpec
---@field card_filter fun(self: ViewAsSkill, to_select: integer, selected: integer[]): boolean
---@field view_as fun(self: ViewAsSkill, cards: integer[])
---@field pattern string
---@field enabled_at_play fun(self: ViewAsSkill, player: Player): boolean
---@field enabled_at_response fun(self: ViewAsSkill, player: Player): boolean
---@param spec ViewAsSkillSpec
---@return ViewAsSkill
function fk.CreateViewAsSkill(spec)
assert(type(spec.name) == "string")
assert(type(spec.view_as) == "function")
local skill = ViewAsSkill:new(spec.name)
skill.viewAs = spec.view_as
if spec.card_filter then
skill.cardFilter = spec.card_filter
end
if type(spec.pattern) == "string" then
skill.pattern = spec.pattern
end
if type(spec.enabled_at_play) == "function" then
skill.enabledAtPlay = spec.enabled_at_play
end
if type(spec.enabled_at_response) == "function" then
skill.enabledAtResponse = spec.enabled_at_response
end
return skill
end
---@class DistanceSpec: SkillSpec ---@class DistanceSpec: SkillSpec
---@field correct_func fun(self: DistanceSkill, from: Player, to: Player) ---@field correct_func fun(self: DistanceSkill, from: Player, to: Player)
---@field global boolean ---@field global boolean

View File

@ -20,6 +20,7 @@ Engine = require "core.engine"
Package = require "core.package" Package = require "core.package"
General = require "core.general" General = require "core.general"
Card = require "core.card" Card = require "core.card"
Exppattern = require "core.exppattern"
Skill = require "core.skill" Skill = require "core.skill"
Player = require "core.player" Player = require "core.player"

View File

@ -55,9 +55,15 @@ function Room:initialize(_room)
end end
local co = coroutine.create(co_func) local co = coroutine.create(co_func)
while not self.game_finished do while not self.game_finished do
coroutine.resume(co) local ret, err_msg = coroutine.resume(co)
-- handle error
if ret == false then
fk.qCritical(err_msg)
print(debug.traceback(co))
break
end
end end
fk.qInfo("Game Finished.")
end end
self.players = {} self.players = {}
@ -426,6 +432,19 @@ function Room:sendLogEvent(type, data, players)
self:doBroadcastNotify("LogEvent", json.encode(data), players) self:doBroadcastNotify("LogEvent", json.encode(data), players)
end end
---@param player ServerPlayer
---@param skill_name string
---@param skill_type nil
function Room:notifySkillInvoked(player, skill_name, skill_type)
self:sendLog{
type = "#InvokeSkill",
from = player.id,
arg = skill_name,
}
-- TODO: notifySkill animation
end
------------------------------------------------------------------------ ------------------------------------------------------------------------
-- interactive functions -- interactive functions
------------------------------------------------------------------------ ------------------------------------------------------------------------
@ -604,64 +623,101 @@ function Room:askForSkillInvoke(player, skill_name, data)
end end
---@param player ServerPlayer ---@param player ServerPlayer
---@param data string
---@return CardUseStruct ---@return CardUseStruct
function Room:askForUseCard(player, card_name, prompt, cancelable, extra_data) function Room:handleUseCardReply(player, data)
data = json.decode(data)
local card = data.card
local targets = data.targets
if type(card) == "string" then
local card_data = json.decode(card)
local skill = Fk.skills[card_data.skill]
local selected_cards = card_data.subcards
if skill:isInstanceOf(ActiveSkill) then
self:notifySkillInvoked(player, skill.name)
skill:onEffect(self, {
from = player.id,
cards = selected_cards,
tos = targets,
})
return nil
elseif skill:isInstanceOf(ViewAsSkill) then
local c = skill:viewAs(selected_cards)
if c then
self:notifySkillInvoked(player, skill.name)
local use = {} ---@type CardUseStruct
use.from = player.id
use.tos = {}
for _, target in ipairs(targets) do
table.insert(use.tos, { target })
end
use.card = c
return use
end
end
else
local use = {} ---@type CardUseStruct
use.from = player.id
use.tos = {}
for _, target in ipairs(targets) do
table.insert(use.tos, { target })
end
if #use.tos == 0 then
use.tos = nil
end
use.card = Fk:getCardById(card)
return use
end
end
-- available extra_data:
-- * must_targets: integer[]
---@param player ServerPlayer
---@param card_name string
---@param pattern string
---@param prompt string
---@return CardUseStruct
function Room:askForUseCard(player, card_name, pattern, prompt, cancelable, extra_data)
local command = "AskForUseCard" local command = "AskForUseCard"
self:notifyMoveFocus(player, card_name) self:notifyMoveFocus(player, card_name)
cancelable = cancelable or false cancelable = cancelable or false
extra_data = extra_data or {} extra_data = extra_data or {}
pattern = pattern or card_name
prompt = prompt or "#AskForUseCard" prompt = prompt or "#AskForUseCard"
local data = {card_name, prompt, cancelable, extra_data} local data = {card_name, pattern, prompt, cancelable, extra_data}
local result = self:doRequest(player, command, json.encode(data)) local result = self:doRequest(player, command, json.encode(data))
if result ~= "" then if result ~= "" then
data = json.decode(result) return self:handleUseCardReply(player, result)
local card = data.card
local targets = data.targets
if type(card) == "string" then
-- TODO: ViewAsSkill
return nil
else
local use = {} ---@type CardUseStruct
use.from = player.id
use.tos = {}
for _, target in ipairs(targets) do
table.insert(use.tos, { target })
end
if #use.tos == 0 then
use.tos = nil
end
use.cardId = card
return use
end
end end
return nil return nil
end end
function Room:askForResponse(player, card_name, prompt, cancelable, extra_data) ---@param player ServerPlayer
---@param card_name string
---@param pattern string
---@param prompt string
---@param cancelable string
function Room:askForResponse(player, card_name, pattern, prompt, cancelable, extra_data)
local command = "AskForResponseCard" local command = "AskForResponseCard"
self:notifyMoveFocus(player, card_name) self:notifyMoveFocus(player, card_name)
cancelable = cancelable or false cancelable = cancelable or false
extra_data = extra_data or {} extra_data = extra_data or {}
pattern = pattern or card_name
prompt = prompt or "#AskForResponseCard" prompt = prompt or "#AskForResponseCard"
local data = {card_name, prompt, cancelable, extra_data} local data = {card_name, pattern, prompt, cancelable, extra_data}
local result = self:doRequest(player, command, json.encode(data)) local result = self:doRequest(player, command, json.encode(data))
if result ~= "" then if result ~= "" then
data = json.decode(result) local use = self:handleUseCardReply(player, result)
local card = data.card if use then
local targets = data.targets return use.card
if type(card) == "string" then
-- TODO: ViewAsSkill
return nil
else
return card
end end
end end
return nil return nil
end end
function Room:askForNullification(players, card_name, prompt, cancelable, extra_data) function Room:askForNullification(players, card_name, pattern, prompt, cancelable, extra_data)
if #players == 0 then if #players == 0 then
return nil return nil
end end
@ -671,33 +727,16 @@ function Room:askForNullification(players, card_name, prompt, cancelable, extra_
cancelable = cancelable or false cancelable = cancelable or false
extra_data = extra_data or {} extra_data = extra_data or {}
prompt = prompt or "#AskForUseCard" prompt = prompt or "#AskForUseCard"
pattern = pattern or card_name
self:notifyMoveFocus(self.alive_players, card_name) self:notifyMoveFocus(self.alive_players, card_name)
self:doBroadcastNotify("WaitForNullification", "") self:doBroadcastNotify("WaitForNullification", "")
local data = {card_name, prompt, cancelable, extra_data} local data = {card_name, pattern, prompt, cancelable, extra_data}
local winner = self:doRaceRequest(command, players, json.encode(data)) local winner = self:doRaceRequest(command, players, json.encode(data))
if winner then if winner then
local result = winner.client_reply local result = winner.client_reply
data = json.decode(result) return self:handleUseCardReply(winner, result)
local card = data.card
local targets = data.targets
if type(card) == "string" then
-- TODO: ViewAsSkill
return nil
else
local use = {} ---@type CardUseStruct
use.from = winner.id
use.tos = {}
for _, target in ipairs(targets) do
table.insert(use.tos, { target })
end
if #use.tos == 0 then
use.tos = nil
end
use.cardId = card
return use
end
end end
return nil return nil
end end
@ -706,6 +745,114 @@ end
-- use card logic, and wrappers -- use card logic, and wrappers
------------------------------------------------------------------------ ------------------------------------------------------------------------
---@param room Room
---@param cardUseEvent CardUseStruct
local sendCardEmotionAndLog = function(room, cardUseEvent)
local from = cardUseEvent.from
local card = cardUseEvent.card
room:setEmotion(room:getPlayerById(from), card.name)
room:doAnimate("Indicate", {
from = from,
to = cardUseEvent.tos or {},
})
local useCardIds = card:isVirtual() and card.subcards or { card.id }
if cardUseEvent.tos and #cardUseEvent.tos > 0 then
local to = {}
for _, t in ipairs(cardUseEvent.tos) do
table.insert(to, t[1])
end
if card:isVirtual() then
if #useCardIds == 0 then
room:sendLog{
type = "#UseV0CardToTargets",
from = from,
to = to,
arg = card:toLogString(),
}
else
room:sendLog{
type = "#UseVCardToTargets",
from = from,
to = to,
card = useCardIds,
arg = card:toLogString(),
}
end
else
room:sendLog{
type = "#UseCardToTargets",
from = from,
to = to,
card = useCardIds
}
end
for _, t in ipairs(cardUseEvent.tos) do
if t[2] then
local temp = {table.unpack(t)}
table.remove(temp, 1)
room:sendLog{
type = "#CardUseCollaborator",
from = t[1],
to = temp,
arg = card.name,
}
end
end
elseif cardUseEvent.toCard then
if card:isVirtual() then
if #useCardIds == 0 then
room:sendLog{
type = "#UseV0CardToCard",
from = from,
arg = cardUseEvent.toCard.name,
arg2 = card:toLogString(),
}
else
room:sendLog{
type = "#UseVCardToCard",
from = from,
card = useCardIds,
arg = cardUseEvent.toCard.name,
arg2 = card:toLogString(),
}
end
else
room:sendLog{
type = "#UseCardToCard",
from = from,
card = useCardIds,
arg = cardUseEvent.toCard.name,
}
end
else
if card:isVirtual() then
if #useCardIds == 0 then
room:sendLog{
type = "#UseV0Card",
from = from,
arg = card:toLogString(),
}
else
room:sendLog{
type = "#UseVCard",
from = from,
card = useCardIds,
arg = card:toLogString(),
}
end
else
room:sendLog{
type = "#UseCard",
from = from,
card = useCardIds,
}
end
end
end
---@param room Room ---@param room Room
---@param cardUseEvent CardUseStruct ---@param cardUseEvent CardUseStruct
---@param aimEventCollaborators table<string, AimStruct[]> ---@param aimEventCollaborators table<string, AimStruct[]>
@ -732,7 +879,7 @@ local onAim = function(room, cardUseEvent, aimEventCollaborators)
if not aimEventCollaborators[toId] or collaboratorsIndex[toId] > #aimEventCollaborators[toId] then if not aimEventCollaborators[toId] or collaboratorsIndex[toId] > #aimEventCollaborators[toId] then
aimStruct = { aimStruct = {
from = cardUseEvent.from, from = cardUseEvent.from,
cardId = cardUseEvent.cardId, card = cardUseEvent.card,
to = toId, to = toId,
targetGroup = cardUseEvent.tos, targetGroup = cardUseEvent.tos,
nullifiedTargets = cardUseEvent.nullifiedTargets or {}, nullifiedTargets = cardUseEvent.nullifiedTargets or {},
@ -760,7 +907,7 @@ local onAim = function(room, cardUseEvent, aimEventCollaborators)
else else
aimStruct = aimEventCollaborators[toId][collaboratorsIndex[toId]] aimStruct = aimEventCollaborators[toId][collaboratorsIndex[toId]]
aimStruct.from = cardUseEvent.from aimStruct.from = cardUseEvent.from
aimStruct.cardId = cardUseEvent.cardId aimStruct.card = cardUseEvent.card
aimStruct.tos = aimGroup aimStruct.tos = aimGroup
aimStruct.targetGroup = cardUseEvent.tos aimStruct.targetGroup = cardUseEvent.tos
aimStruct.nullifiedTargets = cardUseEvent.nullifiedTargets or {} aimStruct.nullifiedTargets = cardUseEvent.nullifiedTargets or {}
@ -820,74 +967,33 @@ end
function Room:useCard(cardUseEvent) function Room:useCard(cardUseEvent)
local from = cardUseEvent.from local from = cardUseEvent.from
self:moveCards({ self:moveCards({
ids = { cardUseEvent.cardId }, ids = self:getSubcardsByRule(cardUseEvent.card),
from = from, from = from,
toArea = Card.Processing, toArea = Card.Processing,
moveReason = fk.ReasonUse, moveReason = fk.ReasonUse,
}) })
if Fk:getCardById(cardUseEvent.cardId).skill then if cardUseEvent.card.skill then
Fk:getCardById(cardUseEvent.cardId).skill:onUse(self, cardUseEvent) cardUseEvent.card.skill:onUse(self, cardUseEvent)
end end
self:setEmotion(self:getPlayerById(from), Fk:getCardById(cardUseEvent.cardId).name) sendCardEmotionAndLog(self, cardUseEvent)
self:doAnimate("Indicate", {
from = from,
to = cardUseEvent.tos or {},
})
if cardUseEvent.tos and #cardUseEvent.tos > 0 then
local to = {}
for _, t in ipairs(cardUseEvent.tos) do
table.insert(to, t[1])
end
self:sendLog{
type = "#UseCardToTargets",
from = from,
to = to,
card = {cardUseEvent.cardId},
}
for _, t in ipairs(cardUseEvent.tos) do
if t[2] then
local temp = {table.unpack(t)}
table.remove(temp, 1)
self:sendLog{
type = "#CardUseCollaborator",
from = t[1],
to = temp,
card = {cardUseEvent.cardId},
}
end
end
elseif cardUseEvent.toCardId then
self:sendLog{
type = "#UseCardToCard",
from = from,
card = {cardUseEvent.cardId},
arg = Fk:getCardById(cardUseEvent.toCardId).name,
}
else
self:sendLog{
type = "#UseCard",
from = from,
card = {cardUseEvent.cardId},
}
end
if self.logic:trigger(fk.PreCardUse, self:getPlayerById(cardUseEvent.from), cardUseEvent) then if self.logic:trigger(fk.PreCardUse, self:getPlayerById(cardUseEvent.from), cardUseEvent) then
return false return false
end end
if not cardUseEvent.extraUse then if not cardUseEvent.extraUse then
self:getPlayerById(cardUseEvent.from):addCardUseHistory(Fk:getCardById(cardUseEvent.cardId).trueName, 1) self:getPlayerById(cardUseEvent.from):addCardUseHistory(cardUseEvent.card.trueName, 1)
end end
if cardUseEvent.responseToEvent then if cardUseEvent.responseToEvent then
cardUseEvent.responseToEvent.cardIdsResponded = cardUseEvent.responseToEvent.cardIdsResponded or {} cardUseEvent.responseToEvent.cardsResponded = cardUseEvent.responseToEvent.cardsResponded or {}
table.insert(cardUseEvent.responseToEvent.cardIdsResponded, cardUseEvent.cardId) table.insert(cardUseEvent.responseToEvent.cardsResponded, cardUseEvent.card)
end end
for _, event in ipairs({ fk.AfterCardUseDeclared, fk.AfterCardTargetDeclared, fk.BeforeCardUseEffect, fk.CardUsing }) do for _, event in ipairs({ fk.AfterCardUseDeclared, fk.AfterCardTargetDeclared, fk.BeforeCardUseEffect, fk.CardUsing }) do
if not cardUseEvent.toCardId and #TargetGroup:getRealTargets(cardUseEvent.tos) == 0 then if not cardUseEvent.toCard and #TargetGroup:getRealTargets(cardUseEvent.tos) == 0 then
break break
end end
@ -899,20 +1005,21 @@ function Room:useCard(cardUseEvent)
break break
end end
if Fk:getCardById(cardUseEvent.cardId).type == Card.TypeEquip then local realCardIds = self:getSubcardsByRule(cardUseEvent.card, { Card.Processing })
if self:getCardArea(cardUseEvent.cardId) ~= Card.Processing then if cardUseEvent.card.type == Card.TypeEquip then
if #realCardIds == 0 then
break break
end end
if self:getPlayerById(TargetGroup:getRealTargets(cardUseEvent.tos)[1]).dead then if self:getPlayerById(TargetGroup:getRealTargets(cardUseEvent.tos)[1]).dead then
self.moveCards({ self.moveCards({
ids = { cardUseEvent.cardId }, ids = realCardIds,
toArea = Card.DiscardPile, toArea = Card.DiscardPile,
moveReason = fk.ReasonPutIntoDiscardPile, moveReason = fk.ReasonPutIntoDiscardPile,
}) })
else else
local target = TargetGroup:getRealTargets(cardUseEvent.tos)[1] local target = TargetGroup:getRealTargets(cardUseEvent.tos)[1]
local existingEquipId = self:getPlayerById(target):getEquipment(Fk:getCardById(cardUseEvent.cardId).sub_type) local existingEquipId = self:getPlayerById(target):getEquipment(cardUseEvent.card.sub_type)
if existingEquipId then if existingEquipId then
self:moveCards( self:moveCards(
{ {
@ -922,7 +1029,7 @@ function Room:useCard(cardUseEvent)
moveReason = fk.ReasonPutIntoDiscardPile, moveReason = fk.ReasonPutIntoDiscardPile,
}, },
{ {
ids = { cardUseEvent.cardId }, ids = realCardIds,
to = target, to = target,
toArea = Card.PlayerEquip, toArea = Card.PlayerEquip,
moveReason = fk.ReasonUse, moveReason = fk.ReasonUse,
@ -930,7 +1037,7 @@ function Room:useCard(cardUseEvent)
) )
else else
self:moveCards({ self:moveCards({
ids = { cardUseEvent.cardId }, ids = realCardIds,
to = target, to = target,
toArea = Card.PlayerEquip, toArea = Card.PlayerEquip,
moveReason = fk.ReasonUse, moveReason = fk.ReasonUse,
@ -939,8 +1046,8 @@ function Room:useCard(cardUseEvent)
end end
break break
elseif Fk:getCardById(cardUseEvent.cardId).sub_type == Card.SubtypeDelayedTrick then elseif cardUseEvent.card.sub_type == Card.SubtypeDelayedTrick then
if self:getCardArea(cardUseEvent.cardId) ~= Card.Processing then if #realCardIds == 0 then
break break
end end
@ -948,14 +1055,14 @@ function Room:useCard(cardUseEvent)
if not self:getPlayerById(target).dead then if not self:getPlayerById(target).dead then
local findSameCard = false local findSameCard = false
for _, cardId in ipairs(self:getPlayerById(target):getCardIds(Player.Judge)) do for _, cardId in ipairs(self:getPlayerById(target):getCardIds(Player.Judge)) do
if Fk:getCardById(cardId).trueName == Fk:getCardById(cardUseEvent.cardId) then if Fk:getCardById(cardId).trueName == cardUseEvent.card.trueName then
findSameCard = true findSameCard = true
end end
end end
if not findSameCard then if not findSameCard then
self:moveCards({ self:moveCards({
ids = { cardUseEvent.cardId }, ids = realCardIds,
to = target, to = target,
toArea = Card.PlayerJudge, toArea = Card.PlayerJudge,
moveReason = fk.ReasonUse, moveReason = fk.ReasonUse,
@ -966,7 +1073,7 @@ function Room:useCard(cardUseEvent)
end end
self:moveCards({ self:moveCards({
ids = { cardUseEvent.cardId }, ids = realCardIds,
toArea = Card.DiscardPile, toArea = Card.DiscardPile,
moveReason = fk.ReasonPutIntoDiscardPile, moveReason = fk.ReasonPutIntoDiscardPile,
}) })
@ -974,13 +1081,13 @@ function Room:useCard(cardUseEvent)
break break
end end
if Fk:getCardById(cardUseEvent.cardId).skill then if cardUseEvent.card.skill then
---@type CardEffectEvent ---@type CardEffectEvent
local cardEffectEvent = { local cardEffectEvent = {
from = cardUseEvent.from, from = cardUseEvent.from,
tos = cardUseEvent.tos, tos = cardUseEvent.tos,
cardId = cardUseEvent.cardId, card = cardUseEvent.card,
toCardId = cardUseEvent.toCardId, toCard = cardUseEvent.toCard,
responseToEvent = cardUseEvent.responseToEvent, responseToEvent = cardUseEvent.responseToEvent,
nullifiedTargets = cardUseEvent.nullifiedTargets, nullifiedTargets = cardUseEvent.nullifiedTargets,
disresponsiveList = cardUseEvent.disresponsiveList, disresponsiveList = cardUseEvent.disresponsiveList,
@ -989,7 +1096,7 @@ function Room:useCard(cardUseEvent)
cardIdsResponded = cardUseEvent.nullifiedTargets, cardIdsResponded = cardUseEvent.nullifiedTargets,
} }
if cardUseEvent.toCardId ~= nil then if cardUseEvent.toCard ~= nil then
self:doCardEffect(cardEffectEvent) self:doCardEffect(cardEffectEvent)
else else
local collaboratorsIndex = {} local collaboratorsIndex = {}
@ -1034,9 +1141,11 @@ function Room:useCard(cardUseEvent)
end end
self.logic:trigger(fk.CardUseFinished, self:getPlayerById(cardUseEvent.from), cardUseEvent) self.logic:trigger(fk.CardUseFinished, self:getPlayerById(cardUseEvent.from), cardUseEvent)
if self:getCardArea(cardUseEvent.cardId) == Card.Processing then
local leftRealCardIds = self:getSubcardsByRule(cardUseEvent.card, { Card.Processing })
if #leftRealCardIds > 0 then
self:moveCards({ self:moveCards({
ids = { cardUseEvent.cardId }, ids = leftRealCardIds,
toArea = Card.DiscardPile, toArea = Card.DiscardPile,
moveReason = fk.ReasonPutIntoDiscardPile, moveReason = fk.ReasonPutIntoDiscardPile,
}) })
@ -1053,7 +1162,7 @@ function Room:doCardEffect(cardEffectEvent)
break break
end end
if not cardEffectEvent.toCardId and (not (self:getPlayerById(cardEffectEvent.to):isAlive() and cardEffectEvent.to) or #self:deadPlayerFilter(TargetGroup:getRealTargets(cardEffectEvent.tos)) == 0) then if not cardEffectEvent.toCard and (not (self:getPlayerById(cardEffectEvent.to):isAlive() and cardEffectEvent.to) or #self:deadPlayerFilter(TargetGroup:getRealTargets(cardEffectEvent.tos)) == 0) then
break break
end end
@ -1066,7 +1175,7 @@ function Room:doCardEffect(cardEffectEvent)
end end
if event == fk.PreCardEffect then if event == fk.PreCardEffect then
if Fk:getCardById(cardEffectEvent.cardId).name == 'slash' and if cardEffectEvent.card.name == 'slash' and
not ( not (
cardEffectEvent.disresponsive or cardEffectEvent.disresponsive or
cardEffectEvent.unoffsetable or cardEffectEvent.unoffsetable or
@ -1076,11 +1185,11 @@ function Room:doCardEffect(cardEffectEvent)
local to = self:getPlayerById(cardEffectEvent.to) local to = self:getPlayerById(cardEffectEvent.to)
local use = self:askForUseCard(to, "jink") local use = self:askForUseCard(to, "jink")
if use then if use then
use.toCardId = cardEffectEvent.cardId use.toCard = cardEffectEvent.card
use.responseToEvent = cardEffectEvent use.responseToEvent = cardEffectEvent
self:useCard(use) self:useCard(use)
end end
elseif Fk:getCardById(cardEffectEvent.cardId).type == Card.TypeTrick then elseif cardEffectEvent.card.type == Card.TypeTrick then
local players = {} local players = {}
for _, p in ipairs(self.alive_players) do for _, p in ipairs(self.alive_players) do
local cards = p.player_cards[Player.Hand] local cards = p.player_cards[Player.Hand]
@ -1094,7 +1203,7 @@ function Room:doCardEffect(cardEffectEvent)
local use = self:askForNullification(players) local use = self:askForNullification(players)
if use then if use then
use.toCardId = cardEffectEvent.cardId use.toCard = cardEffectEvent.card
use.responseToEvent = cardEffectEvent use.responseToEvent = cardEffectEvent
self:useCard(use) self:useCard(use)
end end
@ -1102,9 +1211,8 @@ function Room:doCardEffect(cardEffectEvent)
end end
if event == fk.CardEffecting then if event == fk.CardEffecting then
local cardEffecting = Fk:getCardById(cardEffectEvent.cardId) if cardEffectEvent.card.skill then
if cardEffecting.skill then cardEffectEvent.card.skill:onEffect(self, cardEffectEvent)
cardEffecting.skill:onEffect(self, cardEffectEvent)
end end
end end
end end
@ -1113,22 +1221,48 @@ end
---@param cardResponseEvent CardResponseEvent ---@param cardResponseEvent CardResponseEvent
function Room:responseCard(cardResponseEvent) function Room:responseCard(cardResponseEvent)
local from = cardResponseEvent.customFrom or cardResponseEvent.from local from = cardResponseEvent.customFrom or cardResponseEvent.from
local card = cardResponseEvent.card
local cardIds = self:getSubcardsByRule(card)
if card:isVirtual() then
if #cardIds == 0 then
self:sendLog{
type = "#ResponsePlayV0Card",
from = from,
arg = card:toLogString(),
}
else
self:sendLog{
type = "#ResponsePlayVCard",
from = from,
card = cardIds,
arg = card:toLogString(),
}
end
else
self:sendLog{
type = "#ResponsePlayCard",
from = from,
card = cardIds,
}
end
self:moveCards({ self:moveCards({
ids = { cardResponseEvent.cardId }, ids = cardIds,
from = from, from = from,
toArea = Card.Processing, toArea = Card.Processing,
moveReason = fk.ReasonResonpse, moveReason = fk.ReasonResonpse,
}) })
self:setEmotion(self:getPlayerById(from), Fk:getCardById(cardResponseEvent.cardId).name) self:setEmotion(self:getPlayerById(from), card.name)
for _, event in ipairs({ fk.PreCardRespond, fk.CardResponding, fk.CardRespondFinished }) do for _, event in ipairs({ fk.PreCardRespond, fk.CardResponding, fk.CardRespondFinished }) do
self.logic:trigger(event, self:getPlayerById(cardResponseEvent.from), cardResponseEvent) self.logic:trigger(event, self:getPlayerById(cardResponseEvent.from), cardResponseEvent)
end end
if self:getCardArea(cardResponseEvent.cardId) == Card.Processing or cardResponseEvent.skipDrop then local realCardIds = self:getSubcardsByRule(cardResponseEvent.card, { Card.Processing })
if #realCardIds > 0 and not cardResponseEvent.skipDrop then
self:moveCards({ self:moveCards({
ids = { cardResponseEvent.cardId }, ids = realCardIds,
toArea = Card.DiscardPile, toArea = Card.DiscardPile,
moveReason = fk.ReasonPutIntoDiscardPile, moveReason = fk.ReasonPutIntoDiscardPile,
}) })
@ -1780,6 +1914,25 @@ function Room:gameOver(winner)
coroutine.yield() coroutine.yield()
end end
---@param card Card
---@param fromAreas CardArea[]|null
---@return integer[]
function Room:getSubcardsByRule(card, fromAreas)
if card:isVirtual() and #card.subcards == 0 then
return {}
end
local cardIds = {}
fromAreas = fromAreas or {}
for _, cardId in ipairs(card:isVirtual() and card.subcards or { card.id }) do
if #fromAreas == 0 or table.contains(fromAreas, self:getCardArea(cardId)) then
table.insert(cardIds, cardId)
end
end
return cardIds
end
function CreateRoom(_room) function CreateRoom(_room)
RoomInstance = Room:new(_room) RoomInstance = Room:new(_room)
end end

View File

@ -10,13 +10,13 @@
---@alias DyingStruct { who: integer, damage: DamageStruct } ---@alias DyingStruct { who: integer, damage: DamageStruct }
---@alias DeathStruct { who: integer, damage: DamageStruct } ---@alias DeathStruct { who: integer, damage: DamageStruct }
---@alias CardUseStruct { from: integer, tos: TargetGroup, cardId: integer, toCardId: integer|null, responseToEvent: CardUseStruct|null, nullifiedTargets: interger[]|null, extraUse: boolean|null, disresponsiveList: integer[]|null, unoffsetableList: integer[]|null, addtionalDamage: integer|null, customFrom: integer|null, cardIdsResponded: integer[]|null } ---@alias CardUseStruct { from: integer, tos: TargetGroup, card: Card, toCard: Card|null, responseToEvent: CardUseStruct|null, nullifiedTargets: interger[]|null, extraUse: boolean|null, disresponsiveList: integer[]|null, unoffsetableList: integer[]|null, addtionalDamage: integer|null, customFrom: integer|null, cardsResponded: Card[]|null }
---@alias AimStruct { from: integer, cardId: integer, tos: AimGroup, to: integer, subTargets: integer[]|null, targetGroup: TargetGroup|null, nullifiedTargets: integer[]|null, firstTarget: boolean, additionalDamage: integer|null, disresponsive: boolean|null, unoffsetableList: boolean|null } ---@alias AimStruct { from: integer, card: Card, tos: AimGroup, to: integer, subTargets: integer[]|null, targetGroup: TargetGroup|null, nullifiedTargets: integer[]|null, firstTarget: boolean, additionalDamage: integer|null, disresponsive: boolean|null, unoffsetableList: boolean|null }
---@alias CardEffectEvent { from: integer, to: integer, subTargets: integer[]|null, tos: TargetGroup, cardId: integer, toCardId: integer|null, responseToEvent: CardUseStruct|null, nullifiedTargets: interger[]|null, extraUse: boolean|null, disresponsiveList: integer[]|null, unoffsetableList: integer[]|null, addtionalDamage: integer|null, customFrom: integer|null, cardIdsResponded: integer[]|null, disresponsive: boolean|null, unoffsetable: boolean|null } ---@alias CardEffectEvent { from: integer, to: integer, subTargets: integer[]|null, tos: TargetGroup, card: Card, toCard: Card|null, responseToEvent: CardUseStruct|null, nullifiedTargets: interger[]|null, extraUse: boolean|null, disresponsiveList: integer[]|null, unoffsetableList: integer[]|null, addtionalDamage: integer|null, customFrom: integer|null, cardsResponded: Card[]|null, disresponsive: boolean|null, unoffsetable: boolean|null }
---@alias SkillEffectEvent { from: integer, tos: integer[], cards: integer[] } ---@alias SkillEffectEvent { from: integer, tos: integer[], cards: integer[] }
---@alias JudgeStruct { who: ServerPlayer, card: Card, reason: string } ---@alias JudgeStruct { who: ServerPlayer, card: Card, reason: string }
---@alias CardResponseEvent { from: integer, cardId: integer, responseToEvent: CardEffectEvent|null, skipDrop: boolean|null, customFrom: integer|null } ---@alias CardResponseEvent { from: integer, card: Card, responseToEvent: CardEffectEvent|null, skipDrop: boolean|null, customFrom: integer|null }
---@alias CardMoveReason integer ---@alias CardMoveReason integer

View File

@ -125,7 +125,7 @@ GameRule = fk.CreateTriggerSkill{
---@type CardEffectEvent ---@type CardEffectEvent
local effect_data = { local effect_data = {
cardId = cards[i], card = card,
to = player.id, to = player.id,
tos = { {player.id} }, tos = { {player.id} },
} }
@ -144,26 +144,8 @@ GameRule = fk.CreateTriggerSkill{
local result = room:doRequest(player, "PlayCard", player.id) local result = room:doRequest(player, "PlayCard", player.id)
if result == "" then break end if result == "" then break end
local data = json.decode(result) local use = room:handleUseCardReply(player, result)
local card = data.card if use then
local targets = data.targets
if type(card) == "string" then
local card_data = json.decode(card)
local skill = Fk.skills[card_data.skill]
local selected_cards = card_data.subcards
skill:onEffect(room, {
from = player.id,
cards = selected_cards,
tos = targets,
})
else
local use = {} ---@type CardUseStruct
use.from = player.id
use.tos = {}
for _, target in ipairs(targets) do
table.insert(use.tos, { target })
end
use.cardId = card
room:useCard(use) room:useCard(use)
end end
end end

View File

@ -11,6 +11,12 @@ Fk:loadTranslationTable{
["qun"] = "", ["qun"] = "",
} }
Fk:loadTranslationTable{
["black"] = "黑色",
["red"] = '<font color="#CC3131">红色</font>',
["nocolor"] = '<font color="grey">无色</font>',
}
local caocao = General:new(extension, "caocao", "wei", 4) local caocao = General:new(extension, "caocao", "wei", 4)
Fk:loadTranslationTable{ Fk:loadTranslationTable{
["caocao"] = "曹操", ["caocao"] = "曹操",
@ -74,9 +80,27 @@ Fk:loadTranslationTable{
["liubei"] = "刘备", ["liubei"] = "刘备",
} }
local wusheng = fk.CreateViewAsSkill{
name = "wusheng",
pattern = "slash",
card_filter = function(self, to_select, selected)
if #selected == 1 then return false end
return Fk:getCardById(to_select).color == Card.Red
end,
view_as = function(self, cards)
if #cards ~= 1 then
return nil
end
local c = Fk:cloneCard("slash")
c:addSubcard(cards[1])
return c
end,
}
local guanyu = General:new(extension, "guanyu", "shu", 4) local guanyu = General:new(extension, "guanyu", "shu", 4)
guanyu:addSkill(wusheng)
Fk:loadTranslationTable{ Fk:loadTranslationTable{
["guanyu"] = "关羽", ["guanyu"] = "关羽",
["wusheng"] = "武圣",
} }
local zhangfei = General:new(extension, "zhangfei", "shu", 4) local zhangfei = General:new(extension, "zhangfei", "shu", 4)

View File

@ -195,6 +195,7 @@ local dismantlement = fk.CreateTrickCard{
} }
Fk:loadTranslationTable{ Fk:loadTranslationTable{
["dismantlement"] = "过河拆桥", ["dismantlement"] = "过河拆桥",
["dismantlement_skill"] = "过河拆桥",
} }
extension:addCards({ extension:addCards({
@ -241,6 +242,7 @@ local snatch = fk.CreateTrickCard{
} }
Fk:loadTranslationTable{ Fk:loadTranslationTable{
["snatch"] = "顺手牵羊", ["snatch"] = "顺手牵羊",
["snatch_skill"] = "顺手牵羊",
} }
extension:addCards({ extension:addCards({
@ -275,11 +277,11 @@ local duelSkill = fk.CreateActiveSkill{
break break
end end
local cardIdResponded = room:askForResponse(currentResponser, 'slash') local cardResponded = room:askForResponse(currentResponser, 'slash')
if cardIdResponded then if cardResponded then
room:responseCard({ room:responseCard({
from = currentResponser.id, from = currentResponser.id,
cardId = cardIdResponded, card = cardResponded,
responseToEvent = effect, responseToEvent = effect,
}) })
else else
@ -336,17 +338,13 @@ local collateralSkill = fk.CreateActiveSkill{
cardUseEvent.tos = { { cardUseEvent.tos[1][1], cardUseEvent.tos[2][1] } } cardUseEvent.tos = { { cardUseEvent.tos[1][1], cardUseEvent.tos[2][1] } }
end, end,
on_effect = function(self, room, effect) on_effect = function(self, room, effect)
local cardIdResponded = nil local use = room:askForUseCard(
if not (effect.disresponsive or table.contains(effect.disresponsiveList or {}, effect.to)) then room:getPlayerById(effect.to),
cardIdResponded = room:askForResponse(room:getPlayerById(effect.to), 'slash') "slash", nil, nil, nil, { must_targets = effect.subTargets }
end )
if cardIdResponded then if use then
room:useCard({ room:useCard(use)
from = effect.to,
tos = { { effect.subTargets[1] } },
cardId = cardIdResponded,
})
else else
room:obtainCard(effect.from, room:getPlayerById(effect.to):getEquipment(Card.SubtypeWeapon), true, fk.ReasonGive) room:obtainCard(effect.from, room:getPlayerById(effect.to):getEquipment(Card.SubtypeWeapon), true, fk.ReasonGive)
end end
@ -436,15 +434,15 @@ local savageAssaultSkill = fk.CreateActiveSkill{
end end
end, end,
on_effect = function(self, room, effect) on_effect = function(self, room, effect)
local cardIdResponded = nil local cardResponded = nil
if not (effect.disresponsive or table.contains(effect.disresponsiveList or {}, effect.to)) then if not (effect.disresponsive or table.contains(effect.disresponsiveList or {}, effect.to)) then
cardIdResponded = room:askForResponse(room:getPlayerById(effect.to), 'slash') cardResponded = room:askForResponse(room:getPlayerById(effect.to), 'slash')
end end
if cardIdResponded then if cardResponded then
room:responseCard({ room:responseCard({
from = effect.to, from = effect.to,
cardId = cardIdResponded, card = cardResponded,
responseToEvent = effect, responseToEvent = effect,
}) })
else else
@ -485,15 +483,15 @@ local archeryAttackSkill = fk.CreateActiveSkill{
end end
end, end,
on_effect = function(self, room, effect) on_effect = function(self, room, effect)
local cardIdResponded = nil local cardResponded = nil
if not (effect.disresponsive or table.contains(effect.disresponsiveList or {}, effect.to)) then if not (effect.disresponsive or table.contains(effect.disresponsiveList or {}, effect.to)) then
cardIdResponded = room:askForResponse(room:getPlayerById(effect.to), 'jink') cardResponded = room:askForResponse(room:getPlayerById(effect.to), 'jink')
end end
if cardIdResponded then if cardResponded then
room:responseCard({ room:responseCard({
from = effect.to, from = effect.to,
cardId = cardIdResponded, card = cardResponded,
responseToEvent = effect, responseToEvent = effect,
}) })
else else
@ -628,7 +626,7 @@ local lightningSkill = fk.CreateActiveSkill{
local to = room:getPlayerById(effect.to) local to = room:getPlayerById(effect.to)
local nextp = to:getNextAlive() local nextp = to:getNextAlive()
room:moveCards{ room:moveCards{
ids = { effect.cardId }, ids = room:getSubcardsByRule(effect.card, { Card.Processing }),
to = nextp.id, to = nextp.id,
toArea = Card.PlayerJudge, toArea = Card.PlayerJudge,
moveReason = fk.ReasonPut moveReason = fk.ReasonPut
@ -686,7 +684,7 @@ local indulgenceSkill = fk.CreateActiveSkill{
end, end,
on_nullified = function(self, room, effect) on_nullified = function(self, room, effect)
room:moveCards{ room:moveCards{
ids = { effect.cardId }, ids = room:getSubcardsByRule(effect.card, { Card.Processing }),
toArea = Card.DiscardPile, toArea = Card.DiscardPile,
moveReason = fk.ReasonPutIntoDiscardPile moveReason = fk.ReasonPutIntoDiscardPile
} }

View File

@ -26,6 +26,7 @@ Item {
property var selected_targets: [] property var selected_targets: []
property string responding_card property string responding_card
property bool respond_play: false property bool respond_play: false
property var extra_data: ({})
Image { Image {
source: AppPath + "/image/gamebg" source: AppPath + "/image/gamebg"

View File

@ -13,7 +13,6 @@ RowLayout {
property bool selected: selfPhoto.selected property bool selected: selfPhoto.selected
property bool is_pending: false
property string pending_skill: "" property string pending_skill: ""
property var pending_card property var pending_card
property var pendings: [] // int[], store cid property var pendings: [] // int[], store cid
@ -116,7 +115,7 @@ RowLayout {
if (cname) { if (cname) {
let ids = [], cards = handcardAreaItem.cards; let ids = [], cards = handcardAreaItem.cards;
for (let i = 0; i < cards.length; i++) { for (let i = 0; i < cards.length; i++) {
if (cards[i].name === cname) if (JSON.parse(Backend.callLuaFunction("CardFitPattern", [cards[i].name, cname])))
ids.push(cards[i].cid); ids.push(cards[i].cid);
} }
handcardAreaItem.enableCards(ids); handcardAreaItem.enableCards(ids);
@ -242,7 +241,10 @@ RowLayout {
function enableSkills(cname) { function enableSkills(cname) {
if (cname) { if (cname) {
// TODO: vs skill for (let i = 0; i < skillButtons.count; i++) {
let item = skillButtons.itemAt(i);
item.enabled = JSON.parse(Backend.callLuaFunction("SkillFitPattern", [item.orig, cname]));
}
return; return;
} }
for (let i = 0; i < skillButtons.count; i++) { for (let i = 0; i < skillButtons.count; i++) {

View File

@ -298,6 +298,14 @@ function enableTargets(card) { // card: int | { skill: string, subcards: int[] }
okButton.enabled = JSON.parse(Backend.callLuaFunction( okButton.enabled = JSON.parse(Backend.callLuaFunction(
"CardFeasible", [card, selected_targets] "CardFeasible", [card, selected_targets]
)); ));
if (okButton.enabled) {
if (roomScene.extra_data instanceof Object) {
let must = roomScene.extra_data.must_targets;
if (must instanceof Array) {
okButton.enabled = (must.length === 0);
}
}
}
} else { } else {
all_photos.forEach(photo => { all_photos.forEach(photo => {
photo.state = "normal"; photo.state = "normal";
@ -337,6 +345,16 @@ function updateSelectedTargets(playerid, selected) {
okButton.enabled = JSON.parse(Backend.callLuaFunction( okButton.enabled = JSON.parse(Backend.callLuaFunction(
"CardFeasible", [card, selected_targets] "CardFeasible", [card, selected_targets]
)); ));
if (okButton.enabled) {
if (roomScene.extra_data instanceof Object) {
let must = roomScene.extra_data.must_targets;
if (must instanceof Array) {
okButton.enabled = (must.filter((val) => {
return selected_targets.indexOf(val) === -1;
}).length === 0);
}
}
}
} else { } else {
all_photos.forEach(photo => { all_photos.forEach(photo => {
photo.state = "normal"; photo.state = "normal";
@ -628,14 +646,19 @@ callbacks["GameLog"] = function(jsonData) {
} }
callbacks["AskForUseCard"] = function(jsonData) { callbacks["AskForUseCard"] = function(jsonData) {
// jsonData: card, prompt, cancelable, {} // jsonData: card, pattern, prompt, cancelable, {}
let data = JSON.parse(jsonData); let data = JSON.parse(jsonData);
let cardname = data[0]; let cardname = data[0];
let prompt = data[1]; let pattern = data[1];
let prompt = data[2];
let extra_data = data[4];
if (extra_data != null) {
roomScene.extra_data = extra_data;
}
roomScene.promptText = Backend.translate(prompt) roomScene.promptText = Backend.translate(prompt)
.arg(Backend.translate(cardname)); .arg(Backend.translate(cardname));
roomScene.responding_card = cardname; roomScene.responding_card = pattern;
roomScene.respond_play = false; roomScene.respond_play = false;
roomScene.state = "responding"; roomScene.state = "responding";
okButton.enabled = false; okButton.enabled = false;
@ -643,14 +666,15 @@ callbacks["AskForUseCard"] = function(jsonData) {
} }
callbacks["AskForResponseCard"] = function(jsonData) { callbacks["AskForResponseCard"] = function(jsonData) {
// jsonData: card, prompt, cancelable, {} // jsonData: card_name, pattern, prompt, cancelable, {}
let data = JSON.parse(jsonData); let data = JSON.parse(jsonData);
let cardname = data[0]; let cardname = data[0];
let prompt = data[1]; let pattern = data[1];
let prompt = data[2];
roomScene.promptText = Backend.translate(prompt) roomScene.promptText = Backend.translate(prompt)
.arg(Backend.translate(cardname)); .arg(Backend.translate(cardname));
roomScene.responding_card = cardname; roomScene.responding_card = pattern;
roomScene.respond_play = true; roomScene.respond_play = true;
roomScene.state = "responding"; roomScene.state = "responding";
okButton.enabled = false; okButton.enabled = false;