Merge branch 'main' into playermat-xml

This commit is contained in:
Chr1Z93 2024-08-02 10:17:59 +02:00
commit e2d31038f7
52 changed files with 1369 additions and 189 deletions

View File

@ -138,7 +138,7 @@
"Playermat3Green.383d8b", "Playermat3Green.383d8b",
"Playermat4Red.0840d5", "Playermat4Red.0840d5",
"LeadInvestigator.acaa93", "LeadInvestigator.acaa93",
"ArkhamDBDeckImporter.a28140", "DeckImporter.a28140",
"Configuration.03804b", "Configuration.03804b",
"DrawingTool.280086", "DrawingTool.280086",
"PlayAreaImageSwapper.b7b45b", "PlayAreaImageSwapper.b7b45b",
@ -229,7 +229,7 @@
0, 0,
0 0
], ],
"SaveName": "Arkham SCE - 3.9.0", "SaveName": "Arkham SCE - 3.9.1",
"Sky": "Sky_Museum", "Sky": "Sky_Museum",
"SkyURL": "https://i.imgur.com/GkQqaOF.jpg", "SkyURL": "https://i.imgur.com/GkQqaOF.jpg",
"SnapPoints_path": "SnapPoints.json", "SnapPoints_path": "SnapPoints.json",

View File

@ -401,6 +401,7 @@
"WarningShot.ec38db", "WarningShot.ec38db",
"TheHomeFront.b80459", "TheHomeFront.b80459",
"JennysTwin45s.d87128", "JennysTwin45s.d87128",
"JennysTwin45s.d87129",
"TokenofFaith.2ea0d0", "TokenofFaith.2ea0d0",
"MistsofRlyeh.5558f1", "MistsofRlyeh.5558f1",
"Shortcut.d4fd4a", "Shortcut.d4fd4a",
@ -569,6 +570,7 @@
"TennesseeSourMash3.b5e5f1", "TennesseeSourMash3.b5e5f1",
"TheBellTolls.6cbc01", "TheBellTolls.6cbc01",
"SearchingforIzzie.426d28", "SearchingforIzzie.426d28",
"SearchingforIzzie.426d29",
"StunningBlow.58c435", "StunningBlow.58c435",
"SharpVision1.4d9a97", "SharpVision1.4d9a97",
"Letmehandlethis.36c0cb", "Letmehandlethis.36c0cb",
@ -1524,6 +1526,9 @@
"NormanWithers.a5d9bb", "NormanWithers.a5d9bb",
"NormanWithers.e0a155", "NormanWithers.e0a155",
"JennyBarnes.9058d3", "JennyBarnes.9058d3",
"JennyBarnesParallel.9058d4",
"JennyBarnesParallelBack.9058d5",
"JennyBarnesParallelFront.9058d6",
"CarolynFern.b03b12", "CarolynFern.b03b12",
"DexterDrake.e015f8", "DexterDrake.e015f8",
"SilasMarsh.3f92cf", "SilasMarsh.3f92cf",

View File

@ -6,5 +6,6 @@
"traits": "Pact.", "traits": "Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"cycle": "The Forgotten Age" "cycle": "The Forgotten Age"
} }

View File

@ -5,5 +5,6 @@
"traits": "Curse.", "traits": "Curse.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"cycle": "The Forgotten Age" "cycle": "The Forgotten Age"
} }

View File

@ -5,5 +5,6 @@
"traits": "Monster. Geist.", "traits": "Monster. Geist.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"classRestriction": "Mystic",
"cycle": "The Scarlet Keys" "cycle": "The Scarlet Keys"
} }

View File

@ -0,0 +1,12 @@
{
"id": "02003-p",
"type": "Investigator",
"class": "Rogue",
"traits": "Drifter. Socialite.",
"willpowerIcons": 3,
"intellectIcons": 3,
"combatIcons": 3,
"agilityIcons": 3,
"cycle": "Pistols and Pearls",
"extraToken": "Reaction"
}

View File

@ -0,0 +1,62 @@
{
"AltLookAngle": {
"x": 0,
"y": 0,
"z": 0
},
"Autoraise": true,
"CardID": 15000,
"ColorDiffuse": {
"b": 0.71324,
"g": 0.71324,
"r": 0.71324
},
"CustomDeck": {
"150": {
"BackIsHidden": true,
"BackURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803235/A5A41E0B1229028C82F0451ED9DA1A97D11030C6/",
"FaceURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803373/86292022002FE355597BCB4D81D3692AEFA1207D/",
"NumHeight": 1,
"NumWidth": 1,
"Type": 0,
"UniqueBack": false
}
},
"Description": "The Dilettante",
"DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/JennyBarnesParallel.9058d4.gmnotes",
"GUID": "9058d4",
"Grid": true,
"GridProjection": false,
"Hands": true,
"HideWhenFaceDown": false,
"IgnoreFoW": false,
"LayoutGroupSortIndex": 0,
"Locked": false,
"LuaScript": "",
"LuaScriptState": "",
"MeasureMovement": false,
"Name": "CardCustom",
"Nickname": "Jenny Barnes (Parallel)",
"SidewaysCard": true,
"Snap": true,
"Sticky": true,
"Tags": [
"Investigator",
"PlayerCard"
],
"Tooltip": true,
"Transform": {
"posX": 26.925,
"posY": 3.688,
"posZ": -2.775,
"rotX": 0,
"rotY": 270,
"rotZ": 0,
"scaleX": 1.15,
"scaleY": 1,
"scaleZ": 1.15
},
"Value": 0,
"XmlUI": ""
}

View File

@ -0,0 +1,12 @@
{
"id": "02003-pb",
"type": "Investigator",
"class": "Rogue",
"traits": "Drifter.",
"willpowerIcons": 3,
"intellectIcons": 3,
"combatIcons": 3,
"agilityIcons": 3,
"cycle": "Pistols and Pearls",
"extraToken": "None"
}

View File

@ -0,0 +1,62 @@
{
"AltLookAngle": {
"x": 0,
"y": 0,
"z": 0
},
"Autoraise": true,
"CardID": 15100,
"ColorDiffuse": {
"b": 0.71324,
"g": 0.71324,
"r": 0.71324
},
"CustomDeck": {
"151": {
"BackIsHidden": true,
"BackURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803235/A5A41E0B1229028C82F0451ED9DA1A97D11030C6/",
"FaceURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803104/B281F0E5830527F2B8C733BE48C7D9FB7465965E/",
"NumHeight": 1,
"NumWidth": 1,
"Type": 0,
"UniqueBack": false
}
},
"Description": "The Dilettante",
"DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/JennyBarnesParallelBack.9058d5.gmnotes",
"GUID": "9058d5",
"Grid": true,
"GridProjection": false,
"Hands": true,
"HideWhenFaceDown": false,
"IgnoreFoW": false,
"LayoutGroupSortIndex": 0,
"Locked": false,
"LuaScript": "",
"LuaScriptState": "",
"MeasureMovement": false,
"Name": "CardCustom",
"Nickname": "Jenny Barnes (Parallel Back)",
"SidewaysCard": true,
"Snap": true,
"Sticky": true,
"Tags": [
"Investigator",
"PlayerCard"
],
"Tooltip": true,
"Transform": {
"posX": 26.925,
"posY": 3.688,
"posZ": -2.775,
"rotX": 0,
"rotY": 270,
"rotZ": 0,
"scaleX": 1.15,
"scaleY": 1,
"scaleZ": 1.15
},
"Value": 0,
"XmlUI": ""
}

View File

@ -0,0 +1,12 @@
{
"id": "02003-pf",
"type": "Investigator",
"class": "Rogue",
"traits": "Drifter. Socialite.",
"willpowerIcons": 3,
"intellectIcons": 3,
"combatIcons": 3,
"agilityIcons": 3,
"cycle": "Pistols and Pearls",
"extraToken": "Reaction"
}

View File

@ -0,0 +1,62 @@
{
"AltLookAngle": {
"x": 0,
"y": 0,
"z": 0
},
"Autoraise": true,
"CardID": 15200,
"ColorDiffuse": {
"b": 0.71324,
"g": 0.71324,
"r": 0.71324
},
"CustomDeck": {
"152": {
"BackIsHidden": true,
"BackURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691802940/2292AE6457CB938D1656262EAAF92C780BE8741C/",
"FaceURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803373/86292022002FE355597BCB4D81D3692AEFA1207D/",
"NumHeight": 1,
"NumWidth": 1,
"Type": 0,
"UniqueBack": false
}
},
"Description": "The Dilettante",
"DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/JennyBarnesParallelFront.9058d6.gmnotes",
"GUID": "9058d6",
"Grid": true,
"GridProjection": false,
"Hands": true,
"HideWhenFaceDown": false,
"IgnoreFoW": false,
"LayoutGroupSortIndex": 0,
"Locked": false,
"LuaScript": "",
"LuaScriptState": "",
"MeasureMovement": false,
"Name": "CardCustom",
"Nickname": "Jenny Barnes (Parallel Front)",
"SidewaysCard": true,
"Snap": true,
"Sticky": true,
"Tags": [
"Investigator",
"PlayerCard"
],
"Tooltip": true,
"Transform": {
"posX": 26.925,
"posY": 3.688,
"posZ": -2.775,
"rotX": 0,
"rotY": 270,
"rotZ": 0,
"scaleX": 1.15,
"scaleY": 1,
"scaleZ": 1.15
},
"Value": 0,
"XmlUI": ""
}

View File

@ -6,5 +6,12 @@
"traits": "Item. Weapon. Firearm.", "traits": "Item. Weapon. Firearm.",
"agilityIcons": 2, "agilityIcons": 2,
"wildIcons": 1, "wildIcons": 1,
"uses": [
{
"count": 0,
"type": "Ammo",
"token": "resource"
}
],
"cycle": "The Dunwich Legacy" "cycle": "The Dunwich Legacy"
} }

View File

@ -22,7 +22,7 @@
"UniqueBack": false "UniqueBack": false
} }
}, },
"Description": "", "Description": "A Perfect Fit",
"DragSelectable": true, "DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/JennysTwin45s.d87128.gmnotes", "GMNotes_path": "AllPlayerCards.15bb07/JennysTwin45s.d87128.gmnotes",
"GUID": "d87128", "GUID": "d87128",

View File

@ -0,0 +1,17 @@
{
"id": "90085",
"type": "Asset",
"slot": "Hand x2",
"class": "Neutral",
"traits": "Item. Weapon. Firearm.",
"agilityIcons": 1,
"wildIcons": 2,
"uses": [
{
"count": 0,
"type": "Ammo",
"token": "resource"
}
],
"cycle": "Pistols and Pearls"
}

View File

@ -0,0 +1,62 @@
{
"AltLookAngle": {
"x": 0,
"y": 0,
"z": 0
},
"Autoraise": true,
"CardID": 15300,
"ColorDiffuse": {
"b": 0.71324,
"g": 0.71324,
"r": 0.71324
},
"CustomDeck": {
"153": {
"BackIsHidden": true,
"BackURL": "https://steamusercontent-a.akamaihd.net/ugc/2342503777940352139/A2D42E7E5C43D045D72CE5CFC907E4F886C8C690/",
"FaceURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803450/5C647BCF9EBE08572D746C439A543A18C4BE2DD9/",
"NumHeight": 1,
"NumWidth": 1,
"Type": 0,
"UniqueBack": false
}
},
"Description": "A Perfect Fit (Advanced)",
"DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/JennysTwin45s.d87129.gmnotes",
"GUID": "d87129",
"Grid": true,
"GridProjection": false,
"Hands": true,
"HideWhenFaceDown": true,
"IgnoreFoW": false,
"LayoutGroupSortIndex": 0,
"Locked": false,
"LuaScript": "",
"LuaScriptState": "",
"MeasureMovement": false,
"Name": "CardCustom",
"Nickname": "Jenny's Twin .45s",
"SidewaysCard": false,
"Snap": true,
"Sticky": true,
"Tags": [
"Asset",
"PlayerCard"
],
"Tooltip": true,
"Transform": {
"posX": 9.005,
"posY": 3.859,
"posZ": -16.695,
"rotX": 1,
"rotY": 270,
"rotZ": 0,
"scaleX": 1,
"scaleY": 1,
"scaleZ": 1
},
"Value": 0,
"XmlUI": ""
}

View File

@ -5,5 +5,6 @@
"traits": "Monster. Shoggoth.", "traits": "Monster. Shoggoth.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"classRestriction": "Guardian",
"cycle": "The Scarlet Keys" "cycle": "The Scarlet Keys"
} }

View File

@ -5,5 +5,6 @@
"traits": "Pact.", "traits": "Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"cycle": "Return to the Forgotten Age" "cycle": "Return to the Forgotten Age"
} }

View File

@ -6,5 +6,6 @@
"traits": "Pact.", "traits": "Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"classRestriction": "Rogue",
"cycle": "The Scarlet Keys" "cycle": "The Scarlet Keys"
} }

View File

@ -6,5 +6,6 @@
"traits": "Paradox.", "traits": "Paradox.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"classRestriction": "Seeker",
"cycle": "The Scarlet Keys" "cycle": "The Scarlet Keys"
} }

View File

@ -0,0 +1,8 @@
{
"id": "90086",
"type": "Treachery",
"class": "Neutral",
"traits": "Task.",
"weakness": true,
"cycle": "Pistols and Pearls"
}

View File

@ -0,0 +1,61 @@
{
"AltLookAngle": {
"x": 0,
"y": 0,
"z": 0
},
"Autoraise": true,
"CardID": 15400,
"ColorDiffuse": {
"b": 0.71324,
"g": 0.71324,
"r": 0.71324
},
"CustomDeck": {
"154": {
"BackIsHidden": true,
"BackURL": "https://steamusercontent-a.akamaihd.net/ugc/2342503777940352139/A2D42E7E5C43D045D72CE5CFC907E4F886C8C690/",
"FaceURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691803586/22C0433DD0E36AF34DA9A08F93EC43A00F0E2EBE/",
"NumHeight": 1,
"NumWidth": 1,
"Type": 0,
"UniqueBack": false
}
},
"Description": "Advanced",
"DragSelectable": true,
"GMNotes_path": "AllPlayerCards.15bb07/SearchingforIzzie.426d29.gmnotes",
"GUID": "426d29",
"Grid": true,
"GridProjection": false,
"Hands": true,
"HideWhenFaceDown": true,
"IgnoreFoW": false,
"LayoutGroupSortIndex": 0,
"Locked": false,
"LuaScript": "",
"LuaScriptState": "",
"MeasureMovement": false,
"Name": "CardCustom",
"Nickname": "Searching for Izzie",
"SidewaysCard": false,
"Snap": true,
"Sticky": true,
"Tags": [
"PlayerCard"
],
"Tooltip": true,
"Transform": {
"posX": 9.074,
"posY": 3.685,
"posZ": -16.71,
"rotX": 0,
"rotY": 270,
"rotZ": 0,
"scaleX": 1,
"scaleY": 1,
"scaleZ": 1
},
"Value": 0,
"XmlUI": ""
}

View File

@ -6,5 +6,6 @@
"traits": "Blunder.", "traits": "Blunder.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"classRestriction": "Survivor",
"cycle": "The Scarlet Keys" "cycle": "The Scarlet Keys"
} }

View File

@ -5,6 +5,7 @@
"traits": "Madness. Pact.", "traits": "Madness. Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"hidden": true, "hidden": true,
"cycle": "Return to the Path to Carcosa" "cycle": "Return to the Path to Carcosa"
} }

View File

@ -5,6 +5,7 @@
"traits": "Madness. Pact.", "traits": "Madness. Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"hidden": true, "hidden": true,
"cycle": "Return to the Path to Carcosa" "cycle": "Return to the Path to Carcosa"
} }

View File

@ -5,6 +5,7 @@
"traits": "Madness. Pact.", "traits": "Madness. Pact.",
"weakness": true, "weakness": true,
"basicWeaknessCount": 1, "basicWeaknessCount": 1,
"modeRestriction": "Campaign",
"hidden": true, "hidden": true,
"cycle": "Return to the Path to Carcosa" "cycle": "Return to the Path to Carcosa"
} }

View File

@ -1 +0,0 @@
{"greenDeck":"","investigators":true,"loadNewest":true,"orangeDeck":"","privateDeck":true,"redDeck":"","whiteDeck":""}

View File

@ -1 +1,95 @@
{"ml":{"0d6da1":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":-28.014},"rot":{"x":0,"y":270.0014,"z":0}},"451eaa":{"lock":false,"pos":{"x":12.252,"y":1.4815,"z":11.986},"rot":{"x":0,"y":269.9999,"z":0}},"5302f2":{"lock":false,"pos":{"x":12.2505,"y":1.4815,"z":-20.0137},"rot":{"x":0,"y":270.0014,"z":0}},"72ab92":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":19.986},"rot":{"x":0,"y":269.9999,"z":0}},"9e73fa":{"lock":false,"pos":{"x":12.2501,"y":1.4815,"z":-12.0137},"rot":{"x":0,"y":269.9998,"z":0}},"cc7eb3":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":3.986},"rot":{"x":0,"y":269.9999,"z":0}},"e2dd57":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":-4.014},"rot":{"x":0,"y":270,"z":0}}}} {
"ml": {
"0d6da1": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": -28.014
},
"rot": {
"x": 0,
"y": 270.0014,
"z": 0
}
},
"451eaa": {
"lock": false,
"pos": {
"x": 12.252,
"y": 1.4815,
"z": 11.986
},
"rot": {
"x": 0,
"y": 269.9999,
"z": 0
}
},
"5302f2": {
"lock": false,
"pos": {
"x": 12.2505,
"y": 1.4815,
"z": -20.0137
},
"rot": {
"x": 0,
"y": 270.0014,
"z": 0
}
},
"72ab92": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": 19.986
},
"rot": {
"x": 0,
"y": 269.9999,
"z": 0
}
},
"9e73fa": {
"lock": false,
"pos": {
"x": 12.2501,
"y": 1.4815,
"z": -12.0137
},
"rot": {
"x": 0,
"y": 269.9998,
"z": 0
}
},
"cc7eb3": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": 3.986
},
"rot": {
"x": 0,
"y": 269.9999,
"z": 0
}
},
"e2dd57": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": -4.014
},
"rot": {
"x": 0,
"y": 270,
"z": 0
}
}
}
}

View File

@ -19,7 +19,7 @@
}, },
"ImageScalar": 1, "ImageScalar": 1,
"ImageSecondaryURL": "", "ImageSecondaryURL": "",
"ImageURL": "https://i.imgur.com/wDp1Woo.jpg", "ImageURL": "https://steamusercontent-a.akamaihd.net/ugc/2466368617691603054/ABA4AB3A811107629323CDA2765871EB36626242/",
"WidthScale": 0 "WidthScale": 0
}, },
"Description": "", "Description": "",
@ -34,10 +34,10 @@
"LayoutGroupSortIndex": 0, "LayoutGroupSortIndex": 0,
"Locked": true, "Locked": true,
"LuaScript": "require(\"arkhamdb/DeckImporter\")", "LuaScript": "require(\"arkhamdb/DeckImporter\")",
"LuaScriptState_path": "ArkhamDBDeckImporter.a28140.luascriptstate", "LuaScriptState_path": "DeckImporter.a28140.luascriptstate",
"MeasureMovement": false, "MeasureMovement": false,
"Name": "Custom_Tile", "Name": "Custom_Tile",
"Nickname": "ArkhamDB Deck Importer", "Nickname": "Deck Importer",
"Snap": false, "Snap": false,
"Sticky": true, "Sticky": true,
"Tooltip": false, "Tooltip": false,

View File

@ -0,0 +1,9 @@
{
"greenDeck": "",
"loadNewest": true,
"orangeDeck": "",
"privateDeck": true,
"redDeck": "",
"standalone": false,
"whiteDeck": ""
}

View File

@ -1 +1,20 @@
{"cd":{"move":false,"scale":false},"tid":[{"name":"Felt - Grey","url":"https://i.imgur.com/N0O6aqj.jpg"},{"name":"Wood","url":"https://i.imgur.com/iOFFsGh.jpg"},{"name":"Wood 2","url":"https://i.imgur.com/SQ2t01d.jpg"}]} {
"cd": {
"move": false,
"scale": false
},
"tid": [
{
"name": "Felt - Grey",
"url": "https://i.imgur.com/N0O6aqj.jpg"
},
{
"name": "Wood",
"url": "https://i.imgur.com/iOFFsGh.jpg"
},
{
"name": "Wood 2",
"url": "https://i.imgur.com/SQ2t01d.jpg"
}
]
}

View File

@ -1 +1,29 @@
{"acknowledgedUpgradeVersions":[],"chaosTokensGUID":[],"optionPanel":{"cardLanguage":"en","changePlayAreaImage":false,"enableCardHelpers":true,"playAreaConnectionColor":{"a":1,"b":0.4,"g":0.4,"r":0.4},"playAreaConnections":true,"playAreaSnapTags":true,"showAttachmentHelper":false,"showCleanUpHelper":false,"showCYOA":false,"showDisplacementTool":false,"showDrawButton":false,"showHandHelper":false,"showSearchAssistant":false,"showTitleSplash":true,"useClassTexture":true,"useClueClickers":false,"useResourceCounters":"disabled","useSnapTags":true}} {
"acknowledgedUpgradeVersions": [],
"chaosTokensGUID": [],
"optionPanel": {
"cardLanguage": "en",
"changePlayAreaImage": false,
"enableCardHelpers": true,
"playAreaConnectionColor": {
"a": 1,
"b": 0.4,
"g": 0.4,
"r": 0.4
},
"playAreaConnections": true,
"playAreaSnapTags": true,
"showAttachmentHelper": false,
"showCYOA": false,
"showCleanUpHelper": false,
"showDisplacementTool": false,
"showDrawButton": false,
"showHandHelper": false,
"showSearchAssistant": false,
"showTitleSplash": true,
"useClassTexture": true,
"useClueClickers": false,
"useResourceCounters": "disabled",
"useSnapTags": true
}
}

View File

@ -1 +1,68 @@
{"claims":{"Black":[],"Blue":[],"Brown":[],"Green":[],"Grey":[],"Orange":[],"Pink":[],"Purple":[],"Red":[],"Teal":[],"White":[],"Yellow":[]},"distance":[],"pitch":[],"visibility":{"Black":{"full":false,"play":false},"Blue":{"full":false,"play":false},"Brown":{"full":false,"play":false},"Green":{"full":false,"play":false},"Grey":{"full":false,"play":false},"Orange":{"full":false,"play":false},"Pink":{"full":false,"play":false},"Purple":{"full":false,"play":false},"Red":{"full":false,"play":false},"Teal":{"full":false,"play":false},"White":{"full":false,"play":false},"Yellow":{"full":false,"play":false}}} {
"claims": {
"Black": [],
"Blue": [],
"Brown": [],
"Green": [],
"Grey": [],
"Orange": [],
"Pink": [],
"Purple": [],
"Red": [],
"Teal": [],
"White": [],
"Yellow": []
},
"distance": [],
"pitch": [],
"visibility": {
"Black": {
"full": false,
"play": false
},
"Blue": {
"full": false,
"play": false
},
"Brown": {
"full": false,
"play": false
},
"Green": {
"full": false,
"play": false
},
"Grey": {
"full": false,
"play": false
},
"Orange": {
"full": false,
"play": false
},
"Pink": {
"full": false,
"play": false
},
"Purple": {
"full": false,
"play": false
},
"Red": {
"full": false,
"play": false
},
"Teal": {
"full": false,
"play": false
},
"White": {
"full": false,
"play": false
},
"Yellow": {
"full": false,
"play": false
}
}
}

View File

@ -1 +1,144 @@
{"checks":[],"decals":[{"locked":false,"name":"Arkham SCE logo","pos":{"x":3.1,"y":2.2},"rotation":0,"scale":{"x":"2","y":"2"},"tooltip":"None","url":"https://steamusercontent-a.akamaihd.net/ugc/2501268517218943111/803E57A7B3E9765DF342050EE6C71D69473A7388/"},{"locked":false,"name":"Bootlegger Finn","pos":{"x":3.5,"y":-1.89},"rotation":"25","scale":{"x":"1","y":"1"},"tooltip":"None","url":"https://steamusercontent-a.akamaihd.net/ugc/2037357792052848566/5DA900C430E97D3DFF2C9B8A3DB1CB2271791FC7/"},{"locked":false,"name":"black bar","pos":{"x":0,"y":-2.7},"rotation":0,"scale":{"x":"8","y":"0.03"},"tooltip":"None","url":"https://steamusercontent-a.akamaihd.net/ugc/2501268517219098388/0936FEE03B410319658B5E05DB5D486CEDDE98F5/"}],"fields":[{"align":3,"array":{"x":"1","y":"1"},"counter":"False","distance":{"x":"1","y":"1"},"fieldColor":{"a":0,"b":1,"g":1,"r":1},"font":"200","locked":false,"name":"Patch Notes","pos":{"x":"0","y":-2.9},"role":"Normal Field","size":{"x":"3750","y":"250"},"textColor":{"a":1,"b":0,"g":0,"r":0},"tooltip":"None","value":["Arkham Horror LCG SCE 3.9.0 - 07/08/2024"]},{"align":2,"array":{"x":"1","y":1},"distance":{"x":"1","y":"1"},"fieldColor":{"a":0,"b":1,"g":1,"r":1},"font":"89","locked":false,"name":"Details","pos":{"x":"0","y":0.4},"role":"Nothing","size":{"x":"3750","y":"2750"},"textColor":{"a":1,"b":0,"g":0,"r":0},"tooltip":"None","value":["Thanks for downloading! We're happy to present you a rather big update this time :-)\n\nNew things\n- updated note card for patch notes (bless Marum for his awesome tool!)\n- automated discarding for Patrice\n- confirmation dialog for discard hotkey (e.g. for locations)\n- helpers for cards that redraw tokens and Kohaku\n- displaying of token count for cards that seal tokens\r\n- new action / ability tokens (replacing the old ones)\r\n- option to enable all card helpers (e.g. Heavy Furs)\r\n- option to load class-colored playermat backgrounds\n- coloring for player names in broadcasts\n- right-click option for RBW button on Player Card Panel to specify trait(s)\n\nUpdates\r\n- performed a small clean up of the bottom corners of the table\n- \"Numpad 9\" to rearranges present tokens (on top of adding a resource)\n- Scroll of Secrets context menu helper now displays player names instead of colors\r\n- Player Card Panel can display fan-made cards with a new \"custom\" cycle button)\n- updated Family Inheritance helper to a proper UI\n- \"Discard object\" gamekey works for selected objects\r\n- updated a bunch of tools like Clean Up Helper, Drawing Tool,\nHand Helper, Token Arranger and Search Assistant\n\nFixes\r\r\n- Bugfix for attempting to draw an encounter card while there is no deck\r\n- Bugfix for Navigation Overlay: now checks if playmat is occupied\r\n- Bugfix for Phase Tracker broadcasting\r\n- Performance and file size improvements (e.g. by adding download\nfunctions for CYOA campaign guides and Arkham Fantasy standees)"]}],"flip":"False","height":"0.1","locks":{"checks":false,"decals":false,"fields":false},"nudgeDistance":0.1,"scale":{"x":"0.3","y":"0.3"},"sheetLocked":true} {
"checks": [],
"decals": [
{
"locked": false,
"name": "Arkham SCE logo",
"pos": {
"x": 3.1,
"y": 2.2
},
"rotation": 0,
"scale": {
"x": "2",
"y": "2"
},
"tooltip": "None",
"url": "https://steamusercontent-a.akamaihd.net/ugc/2501268517218943111/803E57A7B3E9765DF342050EE6C71D69473A7388/"
},
{
"locked": false,
"name": "Bootlegger Finn",
"pos": {
"x": 3.5,
"y": -1.89
},
"rotation": "25",
"scale": {
"x": "1",
"y": "1"
},
"tooltip": "None",
"url": "https://steamusercontent-a.akamaihd.net/ugc/2037357792052848566/5DA900C430E97D3DFF2C9B8A3DB1CB2271791FC7/"
},
{
"locked": false,
"name": "black bar",
"pos": {
"x": 0,
"y": -2.7
},
"rotation": 0,
"scale": {
"x": "8",
"y": "0.03"
},
"tooltip": "None",
"url": "https://steamusercontent-a.akamaihd.net/ugc/2501268517219098388/0936FEE03B410319658B5E05DB5D486CEDDE98F5/"
}
],
"fields": [
{
"align": 3,
"array": {
"x": "1",
"y": "1"
},
"counter": "False",
"distance": {
"x": "1",
"y": "1"
},
"fieldColor": {
"a": 0,
"b": 1,
"g": 1,
"r": 1
},
"font": "200",
"locked": false,
"name": "Patch Notes",
"pos": {
"x": "0",
"y": -2.9
},
"role": "Normal Field",
"size": {
"x": "3750",
"y": "250"
},
"textColor": {
"a": 1,
"b": 0,
"g": 0,
"r": 0
},
"tooltip": "None",
"value": [
"Arkham Horror LCG SCE 3.9.1 - 08/01/2024"
]
},
{
"align": 2,
"array": {
"x": "1",
"y": 1
},
"distance": {
"x": "1",
"y": "1"
},
"fieldColor": {
"a": 0,
"b": 1,
"g": 1,
"r": 1
},
"font": "75",
"locked": false,
"name": "Details",
"pos": {
"x": "0",
"y": 0.4
},
"role": "Nothing",
"size": {
"x": "3750",
"y": "2750"
},
"textColor": {
"a": 1,
"b": 0,
"g": 0,
"r": 0
},
"tooltip": "None",
"value": [
"Minor release notes (3.9.1)\n- Added Parallel Jenny!\r\n- Re-Added Learn to Play PDF to table\n- Fixed bug with \"Remove one Use\" gamekey\n- Updated clue replenishing to locations via hotkey (Numpad 8)\r\r\n- Updated Deck Importer (arkham.build support and better RBW drawing)\n- Updated Search Assistant to properly handle a revealed top card (looking at Norman)\n\nNew things\n- updated note card for patch notes (bless Marum for his awesome tool!)\n- automated discarding for Patrice\n- confirmation dialog for discard hotkey (e.g. for locations)\n- helpers for cards that redraw tokens and Kohaku\n- displaying of token count for cards that seal tokens\r\n- new action / ability tokens (replacing the old ones)\r\n- option to enable all card helpers (e.g. Heavy Furs)\r\n- option to load class-colored playermat backgrounds\n- coloring for player names in broadcasts\n- right-click option for RBW button on Player Card Panel to specify trait(s)\n\nUpdates\r\n- performed a small clean up of the bottom corners of the table\n- \"Numpad 9\" to rearranges present tokens (on top of adding a resource)\n- Scroll of Secrets context menu helper now displays player names instead of colors\r\n- Player Card Panel can display fan-made cards with a new \"custom\" cycle button)\n- updated Family Inheritance helper to a proper UI\n- \"Discard object\" gamekey works for selected objects\r\n- updated a bunch of tools like Clean Up Helper, Drawing Tool,\nHand Helper, Token Arranger and Search Assistant\n\nFixes\r\r\n- Bugfix for attempting to draw an encounter card while there is no deck\r\n- Bugfix for Navigation Overlay: now checks if playmat is occupied\r\n- Bugfix for Phase Tracker broadcasting\r\n- Performance and file size improvements (e.g. by adding download\nfunctions for CYOA campaign guides and Arkham Fantasy standees)"
]
}
],
"flip": "False",
"height": "0.1",
"locks": {
"checks": false,
"decals": false,
"fields": false
},
"nudgeDistance": "0.1",
"scale": {
"x": "0.3",
"y": "0.3"
},
"sheetLocked": true
}

View File

@ -1 +1,10 @@
{"connectionColor":{"a":1,"b":0.4,"g":0.4,"r":0.4},"connectionsEnabled":true,"trackedLocations":[]} {
"connectionColor": {
"a": 1,
"b": 0.4,
"g": 0.4,
"r": 0.4
},
"connectionsEnabled": true,
"trackedLocations": []
}

View File

@ -1 +1,23 @@
{"activeInvestigatorClass":"Neutral","activeInvestigatorId":"00000","isClassTextureEnabled":true,"isDrawButtonVisible":false,"playerColor":"White","slotData":["any","any","any","Tarot","Hand (left)","Hand (right)","Ally","any","any","any","Accessory","Arcane","Arcane","Body"]} {
"activeInvestigatorClass": "Neutral",
"activeInvestigatorId": "00000",
"isClassTextureEnabled": true,
"isDrawButtonVisible": false,
"playerColor": "White",
"slotData": [
"any",
"any",
"any",
"Tarot",
"Hand (left)",
"Hand (right)",
"Ally",
"any",
"any",
"any",
"Accessory",
"Arcane",
"Arcane",
"Body"
]
}

View File

@ -1 +1,23 @@
{"activeInvestigatorClass":"Neutral","activeInvestigatorId":"00000","isClassTextureEnabled":true,"isDrawButtonVisible":false,"playerColor":"Orange","slotData":["any","any","any","Tarot","Hand (left)","Hand (right)","Ally","any","any","any","Accessory","Arcane","Arcane","Body"]} {
"activeInvestigatorClass": "Neutral",
"activeInvestigatorId": "00000",
"isClassTextureEnabled": true,
"isDrawButtonVisible": false,
"playerColor": "Orange",
"slotData": [
"any",
"any",
"any",
"Tarot",
"Hand (left)",
"Hand (right)",
"Ally",
"any",
"any",
"any",
"Accessory",
"Arcane",
"Arcane",
"Body"
]
}

View File

@ -1 +1,23 @@
{"activeInvestigatorClass":"Neutral","activeInvestigatorId":"00000","isClassTextureEnabled":true,"isDrawButtonVisible":false,"playerColor":"Green","slotData":["any","any","any","Tarot","Hand (left)","Hand (right)","Ally","any","any","any","Accessory","Arcane","Arcane","Body"]} {
"activeInvestigatorClass": "Neutral",
"activeInvestigatorId": "00000",
"isClassTextureEnabled": true,
"isDrawButtonVisible": false,
"playerColor": "Green",
"slotData": [
"any",
"any",
"any",
"Tarot",
"Hand (left)",
"Hand (right)",
"Ally",
"any",
"any",
"any",
"Accessory",
"Arcane",
"Arcane",
"Body"
]
}

View File

@ -1 +1,23 @@
{"activeInvestigatorClass":"Neutral","activeInvestigatorId":"00000","isClassTextureEnabled":true,"isDrawButtonVisible":false,"playerColor":"Red","slotData":["any","any","any","Tarot","Hand (left)","Hand (right)","Ally","any","any","any","Accessory","Arcane","Arcane","Body"]} {
"activeInvestigatorClass": "Neutral",
"activeInvestigatorId": "00000",
"isClassTextureEnabled": true,
"isDrawButtonVisible": false,
"playerColor": "Red",
"slotData": [
"any",
"any",
"any",
"Tarot",
"Hand (left)",
"Hand (right)",
"Ally",
"any",
"any",
"any",
"Accessory",
"Arcane",
"Arcane",
"Body"
]
}

View File

@ -1 +1,134 @@
{"ml":{"01d780":{"lock":false,"pos":{"x":12.252,"y":1.4815,"z":11.986},"rot":{"x":0,"y":270.0001,"z":0}},"0dce91":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":-28.014},"rot":{"x":0,"y":269.9792,"z":0}},"23dd51":{"lock":false,"pos":{"x":12.249,"y":1.4815,"z":35.986},"rot":{"x":0,"y":270,"z":0}},"3c4f3c":{"lock":false,"pos":{"x":12.251,"y":1.4815,"z":-20.014},"rot":{"x":0,"y":269.9867,"z":0}},"4c173f":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":3.986},"rot":{"x":0,"y":269.9998,"z":0}},"4dee5a":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":-4.014},"rot":{"x":0,"y":269.9999,"z":0}},"d02940":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":-36.014},"rot":{"x":0,"y":270.0045,"z":0}},"db7039":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":27.986},"rot":{"x":0,"y":270.0001,"z":0}},"ee987d":{"lock":false,"pos":{"x":12.25,"y":1.4815,"z":19.986},"rot":{"x":0,"y":270.0001,"z":0}},"fc7674":{"lock":false,"pos":{"x":12.247,"y":1.4815,"z":-12.016},"rot":{"x":0,"y":270.0001,"z":0}}}} {
"ml": {
"01d780": {
"lock": false,
"pos": {
"x": 12.252,
"y": 1.4815,
"z": 11.986
},
"rot": {
"x": 0,
"y": 270.0001,
"z": 0
}
},
"0dce91": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": -28.014
},
"rot": {
"x": 0,
"y": 269.9792,
"z": 0
}
},
"23dd51": {
"lock": false,
"pos": {
"x": 12.249,
"y": 1.4815,
"z": 35.986
},
"rot": {
"x": 0,
"y": 270,
"z": 0
}
},
"3c4f3c": {
"lock": false,
"pos": {
"x": 12.251,
"y": 1.4815,
"z": -20.014
},
"rot": {
"x": 0,
"y": 269.9867,
"z": 0
}
},
"4c173f": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": 3.986
},
"rot": {
"x": 0,
"y": 269.9998,
"z": 0
}
},
"4dee5a": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": -4.014
},
"rot": {
"x": 0,
"y": 269.9999,
"z": 0
}
},
"d02940": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": -36.014
},
"rot": {
"x": 0,
"y": 270.0045,
"z": 0
}
},
"db7039": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": 27.986
},
"rot": {
"x": 0,
"y": 270.0001,
"z": 0
}
},
"ee987d": {
"lock": false,
"pos": {
"x": 12.25,
"y": 1.4815,
"z": 19.986
},
"rot": {
"x": 0,
"y": 270.0001,
"z": 0
}
},
"fc7674": {
"lock": false,
"pos": {
"x": 12.247,
"y": 1.4815,
"z": -12.016
},
"rot": {
"x": 0,
"y": 270.0001,
"z": 0
}
}
}
}

View File

@ -1 +1,46 @@
{"includeDrawnTokens":true,"percentage":false,"tokenPrecedence":{"":[0,11],"Auto-fail":[-100,7],"Bless":[110,8],"Cultist":[-2,4],"Curse":[-110,9],"Elder Sign":[100,2],"Elder Thing":[-4,6],"Frost":[-105,10],"Skull":[-1,3],"Tablet":[-3,5]}} {
"includeDrawnTokens": true,
"percentage": false,
"tokenPrecedence": {
"": [
0,
11
],
"Auto-fail": [
-100,
7
],
"Bless": [
110,
8
],
"Cultist": [
-2,
4
],
"Curse": [
-110,
9
],
"Elder Sign": [
100,
2
],
"Elder Thing": [
-4,
6
],
"Frost": [
-105,
10
],
"Skull": [
-1,
3
],
"Tablet": [
-3,
5
]
}
}

View File

@ -113,11 +113,25 @@ function startSearch(messageColor, number)
-- get draw deck -- get draw deck
local deckAreaObjects = playermatApi.getDeckAreaObjects(matColor) local deckAreaObjects = playermatApi.getDeckAreaObjects(matColor)
if deckAreaObjects.draw == nil then if deckAreaObjects.draw == nil and deckAreaObjects.topCard == nil then
printToColor(matColor .. " draw deck could not be found!", messageColor, "Red") printToColor(matColor .. " draw deck could not be found!", messageColor, "Red")
return return
end end
-- check for harbinger
local harbinger
if deckAreaObjects.topCard then
harbinger = isHarbinger(deckAreaObjects.topCard.getGMNotes())
elseif deckAreaObjects.draw and not deckAreaObjects.draw.is_face_down then
local cards = deckAreaObjects.draw.getObjects()
harbinger = isHarbinger(cards[#cards].gm_notes)
end
if harbinger then
printToColor("The Harbinger is on top of your deck, searching isn't allowed", messageColor)
return
end
-- get bounds to know the height of the deck -- get bounds to know the height of the deck
local bounds = deckAreaObjects.draw.getBounds() local bounds = deckAreaObjects.draw.getBounds()
drawDeckPosition = bounds.center + Vector(0, bounds.size.y / 2 + 0.2, 0) drawDeckPosition = bounds.center + Vector(0, bounds.size.y / 2 + 0.2, 0)
@ -158,16 +172,21 @@ function startSearch(messageColor, number)
-- handling for Norman Withers -- handling for Norman Withers
if deckAreaObjects.topCard then if deckAreaObjects.topCard then
deckAreaObjects.topCard.setRotation(setAsideRotation) deckAreaObjects.topCard.deal(1, handColor)
number = number - 1
topCardDetected = true topCardDetected = true
end end
searchView() searchView()
Wait.time(function() if number > 0 then
deckAreaObjects = playermatApi.getDeckAreaObjects(matColor)
deckAreaObjects.draw.deal(number, handColor) deckAreaObjects.draw.deal(number, handColor)
end, 1) end
end
function isHarbinger(notes)
local md = JSON.decode(notes or "") or {}
return md.id == "08006"
end end
-- place handCards back into deck and optionally shuffle -- place handCards back into deck and optionally shuffle

View File

@ -40,7 +40,7 @@ do
---@param deckId string ArkhamDB deck id to be loaded ---@param deckId string ArkhamDB deck id to be loaded
---@param isPrivate boolean Whether this deck is published or private on ArkhamDB ---@param isPrivate boolean Whether this deck is published or private on ArkhamDB
---@param loadNewest boolean Whether the newest version of this deck should be loaded ---@param loadNewest boolean Whether the newest version of this deck should be loaded
---@param loadInvestigators boolean Whether investigator cards should be loaded as part of this deck ---@param standalone boolean Whether 'Campaign only' weaknesses should be exluded
---@param callback function Callback which will be sent the results of this load ---@param callback function Callback which will be sent the results of this load
--- Parameters to the callback will be: --- Parameters to the callback will be:
--- slots table A map of card ID to count in the deck --- slots table A map of card ID to count in the deck
@ -54,7 +54,7 @@ do
deckId, deckId,
isPrivate, isPrivate,
loadNewest, loadNewest,
loadInvestigators, standalone,
callback) callback)
-- Get a simple card to see if the bag indexes are complete. If not, abort -- Get a simple card to see if the bag indexes are complete. If not, abort
-- the deck load. The called method will handle player notification. -- the deck load. The called method will handle player notification.
@ -90,7 +90,7 @@ do
return true, json return true, json
end) end)
deck:with(internal.onDeckResult, playerColor, loadNewest, loadInvestigators, callback) deck:with(internal.onDeckResult, playerColor, loadNewest, standalone, callback)
end end
-- Logs that a card could not be loaded in the mod by printing it to the console in the given -- Logs that a card could not be loaded in the mod by printing it to the console in the given
@ -125,7 +125,7 @@ do
---@param deck table ArkhamImportDeck ---@param deck table ArkhamImportDeck
---@param playerColor string Color name of the player mat to place this deck on (e.g. "Red") ---@param playerColor string Color name of the player mat to place this deck on (e.g. "Red")
---@param loadNewest boolean Whether the newest version of this deck should be loaded ---@param loadNewest boolean Whether the newest version of this deck should be loaded
---@param loadInvestigators boolean Whether investigator cards should be loaded as part of this deck ---@param standalone boolean Whether 'Campaign only' weaknesses should be exluded
---@param callback function Callback which will be sent the results of this load. ---@param callback function Callback which will be sent the results of this load.
--- Parameters to the callback will be: --- Parameters to the callback will be:
--- slots table A map of card ID to count in the deck --- slots table A map of card ID to count in the deck
@ -134,7 +134,7 @@ do
--- added from a parent bonded card. --- added from a parent bonded card.
--- customizations table The decoded table of customization upgrades in this deck --- customizations table The decoded table of customization upgrades in this deck
--- playerColor String. Color this deck is being loaded for --- playerColor String. Color this deck is being loaded for
internal.onDeckResult = function(deck, playerColor, loadNewest, loadInvestigators, callback) internal.onDeckResult = function(deck, playerColor, loadNewest, standalone, callback)
-- Load the next deck in the upgrade path if the option is enabled -- Load the next deck in the upgrade path if the option is enabled
if (loadNewest and deck.next_deck ~= nil and deck.next_deck ~= "") then if (loadNewest and deck.next_deck ~= nil and deck.next_deck ~= "") then
buildDeck(playerColor, deck.next_deck) buildDeck(playerColor, deck.next_deck)
@ -147,13 +147,16 @@ do
-- be changed, as later steps may act on cards added in each. For example, a random weakness or -- be changed, as later steps may act on cards added in each. For example, a random weakness or
-- investigator may have bonded cards or taboo entries, and should be present -- investigator may have bonded cards or taboo entries, and should be present
local slots = deck.slots local slots = deck.slots
internal.maybeDrawRandomWeakness(slots, playerColor)
-- get class for investigator to handle specific weaknesses
local class
local card = allCardsBagApi.getCardById(deck.investigator_code)
if card and card.metadata then class = card.metadata.class end
local restrictions = { class = class, standalone = standalone }
internal.maybeDrawRandomWeakness(slots, playerColor, restrictions)
-- handles alternative investigators (parallel, promo or revised art) -- handles alternative investigators (parallel, promo or revised art)
local loadAltInvestigator = "normal" local loadAltInvestigator = internal.addInvestigatorCards(deck, slots)
if loadInvestigators then
loadAltInvestigator = internal.addInvestigatorCards(deck, slots)
end
internal.maybeModifyDeckFromDescription(slots, deck.description_md, playerColor) internal.maybeModifyDeckFromDescription(slots, deck.description_md, playerColor)
internal.maybeAddSummonedServitor(slots) internal.maybeAddSummonedServitor(slots)
@ -179,16 +182,20 @@ do
--- of those cards which will be spawned --- of those cards which will be spawned
---@param playerColor string Color of the player this deck is being loaded for. Used for broadcast ---@param playerColor string Color of the player this deck is being loaded for. Used for broadcast
--- if a weakness is added. --- if a weakness is added.
internal.maybeDrawRandomWeakness = function(slots, playerColor) ---@param restrictions table Additional restrictions:
--- class string Class to restrict weakness to
--- standalone boolean Whether 'Campaign only' weaknesses should be exluded
--- traits? string Trait(s) to use as filter
internal.maybeDrawRandomWeakness = function(slots, playerColor, restrictions)
local randomWeaknessAmount = slots[RANDOM_WEAKNESS_ID] or 0 local randomWeaknessAmount = slots[RANDOM_WEAKNESS_ID] or 0
slots[RANDOM_WEAKNESS_ID] = nil slots[RANDOM_WEAKNESS_ID] = nil
if randomWeaknessAmount > 0 then if randomWeaknessAmount > 0 then
for i = 1, randomWeaknessAmount do local weaknessIds = allCardsBagApi.getRandomWeaknessIds(randomWeaknessAmount, restrictions)
local weaknessId = allCardsBagApi.getRandomWeaknessId() for _, weaknessId in ipairs(weaknessIds) do
slots[weaknessId] = (slots[weaknessId] or 0) + 1 slots[weaknessId] = (slots[weaknessId] or 0) + 1
end end
internal.maybePrint("Added " .. randomWeaknessAmount .. " random basic weakness(es) to deck", playerColor) internal.maybePrint("Added " .. #weaknessIds .. " random basic weakness(es) to deck", playerColor)
end end
end end

View File

@ -21,18 +21,18 @@ local UPGRADED_TOGGLE_LABELS = {}
UPGRADED_TOGGLE_LABELS[true] = "Upgraded" UPGRADED_TOGGLE_LABELS[true] = "Upgraded"
UPGRADED_TOGGLE_LABELS[false] = "Specific" UPGRADED_TOGGLE_LABELS[false] = "Specific"
local LOAD_INVESTIGATOR_TOGGLE_LABELS = {} local STANDALONE_TOGGLE_LABELS = {}
LOAD_INVESTIGATOR_TOGGLE_LABELS[true] = "Yes" STANDALONE_TOGGLE_LABELS[true] = "Yes"
LOAD_INVESTIGATOR_TOGGLE_LABELS[false] = "No" STANDALONE_TOGGLE_LABELS[false] = "No"
local redDeckId = "" redDeckId = ""
local orangeDeckId = "" orangeDeckId = ""
local whiteDeckId = "" whiteDeckId = ""
local greenDeckId = "" greenDeckId = ""
local privateDeck = true local privateDeck = true
local loadNewestDeck = true local loadNewestDeck = true
local loadInvestigators = false local standalone = false
function onLoad(script_state) function onLoad(script_state)
initializeUi(JSON.decode(script_state)) initializeUi(JSON.decode(script_state))
@ -53,7 +53,7 @@ function getUiState()
greenDeck = greenDeckId, greenDeck = greenDeckId,
privateDeck = privateDeck, privateDeck = privateDeck,
loadNewest = loadNewestDeck, loadNewest = loadNewestDeck,
investigators = loadInvestigators standalone = standalone
} }
end end
@ -74,7 +74,7 @@ function initializeUi(savedUiState)
greenDeckId = savedUiState.greenDeck greenDeckId = savedUiState.greenDeck
privateDeck = savedUiState.privateDeck privateDeck = savedUiState.privateDeck
loadNewestDeck = savedUiState.loadNewest loadNewestDeck = savedUiState.loadNewest
loadInvestigators = savedUiState.investigators standalone = savedUiState.standalone
end end
makeOptionToggles() makeOptionToggles()
@ -84,72 +84,74 @@ end
function makeOptionToggles() function makeOptionToggles()
-- common parameters -- common parameters
local checkboxParameters = {} local cParams = {}
checkboxParameters.function_owner = self cParams.function_owner = self
checkboxParameters.width = INPUT_FIELD_WIDTH cParams.width = 1750
checkboxParameters.height = INPUT_FIELD_HEIGHT cParams.height = INPUT_FIELD_HEIGHT
checkboxParameters.scale = { 0.1, 0.1, 0.1 } cParams.position = Vector( 0.22, 0.1, -0.102)
checkboxParameters.font_size = 240 cParams.scale = { 0.1, 0.1, 0.1 }
checkboxParameters.hover_color = { 0.4, 0.6, 0.8 } cParams.font_size = 240
checkboxParameters.color = FIELD_COLOR cParams.hover_color = { 0.4, 0.6, 0.8 }
cParams.color = FIELD_COLOR
-- public / private deck -- public / private deck
checkboxParameters.click_function = "publicPrivateChanged" cParams.click_function = "publishedPrivateChanged"
checkboxParameters.position = { 0.25, 0.1, -0.102 } cParams.tooltip = "Published or private deck?\n\nPLEASE USE A PRIVATE DECK IF JUST FOR TTS TO AVOID FLOODING ARKHAMDB PUBLISHED DECK LISTS!\n\nMake sure to enable deck sharing in your account settings.\n\nKeep this on 'Private' for arkham.build."
checkboxParameters.tooltip = "Published or private deck?\n\nPLEASE USE A PRIVATE DECK IF JUST FOR TTS TO AVOID FLOODING ARKHAMDB PUBLISHED DECK LISTS!" cParams.label = PRIVATE_TOGGLE_LABELS[privateDeck]
checkboxParameters.label = PRIVATE_TOGGLE_LABELS[privateDeck] self.createButton(cParams)
self.createButton(checkboxParameters)
-- load upgraded? -- load upgraded?
checkboxParameters.click_function = "loadUpgradedChanged" cParams.click_function = "loadUpgradedChanged"
checkboxParameters.position = { 0.25, 0.1, -0.01 } cParams.position.z = -0.01
checkboxParameters.tooltip = "Load newest upgrade or exact deck?" cParams.tooltip = "Load newest upgrade or exact deck?"
checkboxParameters.label = UPGRADED_TOGGLE_LABELS[loadNewestDeck] cParams.label = UPGRADED_TOGGLE_LABELS[loadNewestDeck]
self.createButton(checkboxParameters) self.createButton(cParams)
-- load investigators? -- standalone mode?
checkboxParameters.click_function = "loadInvestigatorsChanged" cParams.click_function = "standaloneChanged"
checkboxParameters.position = { 0.25, 0.1, 0.081 } cParams.position.z = 0.081
checkboxParameters.tooltip = "Spawn investigator cards?" cParams.tooltip = "Are you playing standalone mode? Enabling this will make all 'Campaign Only' weaknesses ineligible when determining the random basic weakness(es)."
checkboxParameters.label = LOAD_INVESTIGATOR_TOGGLE_LABELS[loadInvestigators] cParams.label = STANDALONE_TOGGLE_LABELS[standalone]
self.createButton(checkboxParameters) self.createButton(cParams)
end end
-- Create the four deck ID entry fields -- Create the four deck ID entry fields
function makeDeckIdFields() function makeDeckIdFields()
local inputParameters = {} local iParams = {}
-- Parameters common to all entry fields iParams.function_owner = self
inputParameters.function_owner = self iParams.scale = { 0.1, 0.1, 0.1 }
inputParameters.scale = { 0.1, 0.1, 0.1 } iParams.width = INPUT_FIELD_WIDTH
inputParameters.width = INPUT_FIELD_WIDTH iParams.height = INPUT_FIELD_HEIGHT
inputParameters.height = INPUT_FIELD_HEIGHT iParams.font_size = 320
inputParameters.font_size = 320 iParams.tooltip = "Deck ID from ArkhamDB URL of the deck\nPublished URL: 'https://arkhamdb.com/decklist/view/101/knowledge-overwhelming-solo-deck-1.0' = '101'\nPrivate URL: 'https://arkhamdb.com/deck/view/102' = '102'\n\nAlso supports the deck ID from shared decks from arkham.build!"
inputParameters.tooltip = "Deck ID from ArkhamDB URL of the deck\nPublic URL: 'https://arkhamdb.com/decklist/view/101/knowledge-overwhelming-solo-deck-1.0' = '101'\nPrivate URL: 'https://arkhamdb.com/deck/view/102' = '102'\n\nAlso supports the deck ID from shared decks from arkham.build!" iParams.alignment = 3 -- Center
inputParameters.alignment = 3 -- Center iParams.color = FIELD_COLOR
inputParameters.color = FIELD_COLOR iParams.font_color = { 0, 0, 0 }
inputParameters.font_color = { 0, 0, 0 } iParams.validation = 4 -- alphanumeric (to support arkham.build IDs)
inputParameters.validation = 4 -- alphanumeric (to support arkham.build IDs)
-- Green -- Green
inputParameters.input_function = "greenDeckChanged" iParams.input_function = "greenDeckChanged"
inputParameters.position = { -0.166, 0.1, 0.385 } iParams.position = { -0.16, 0.1, 0.385 }
inputParameters.value = greenDeckId iParams.value = greenDeckId
self.createInput(inputParameters) self.createInput(iParams)
-- Red -- Red
inputParameters.input_function = "redDeckChanged" iParams.input_function = "redDeckChanged"
inputParameters.position = { 0.171, 0.1, 0.385 } iParams.position = { 0.165, 0.1, 0.385 }
inputParameters.value = redDeckId iParams.value = redDeckId
self.createInput(inputParameters) self.createInput(iParams)
-- White -- White
inputParameters.input_function = "whiteDeckChanged" iParams.input_function = "whiteDeckChanged"
inputParameters.position = { -0.166, 0.1, 0.474 } iParams.position = { -0.16, 0.1, 0.474 }
inputParameters.value = whiteDeckId iParams.value = whiteDeckId
self.createInput(inputParameters) self.createInput(iParams)
-- Orange -- Orange
inputParameters.input_function = "orangeDeckChanged" iParams.input_function = "orangeDeckChanged"
inputParameters.position = { 0.171, 0.1, 0.474 } iParams.position = { 0.165, 0.1, 0.474 }
inputParameters.value = orangeDeckId iParams.value = orangeDeckId
self.createInput(inputParameters) self.createInput(iParams)
end end
-- Create the Build All button. This is a transparent button which covers the Build All portion of the background graphic -- Create the Build All button. This is a transparent button which covers the Build All portion of the background graphic
@ -172,7 +174,7 @@ function whiteDeckChanged(_, _, inputValue) whiteDeckId = inputValue end
function greenDeckChanged(_, _, inputValue) greenDeckId = inputValue end function greenDeckChanged(_, _, inputValue) greenDeckId = inputValue end
-- Event handlers for toggle buttons -- Event handlers for toggle buttons
function publicPrivateChanged() function publishedPrivateChanged()
privateDeck = not privateDeck privateDeck = not privateDeck
self.editButton({ index = 0, label = PRIVATE_TOGGLE_LABELS[privateDeck] }) self.editButton({ index = 0, label = PRIVATE_TOGGLE_LABELS[privateDeck] })
end end
@ -182,25 +184,37 @@ function loadUpgradedChanged()
self.editButton({ index = 1, label = UPGRADED_TOGGLE_LABELS[loadNewestDeck] }) self.editButton({ index = 1, label = UPGRADED_TOGGLE_LABELS[loadNewestDeck] })
end end
function loadInvestigatorsChanged() function standaloneChanged()
loadInvestigators = not loadInvestigators standalone = not standalone
self.editButton({ index = 2, label = LOAD_INVESTIGATOR_TOGGLE_LABELS[loadInvestigators] }) self.editButton({ index = 2, label = STANDALONE_TOGGLE_LABELS[standalone] })
end end
-- start the deck importing process
function loadDecks() function loadDecks()
co = coroutine.create(loadDecksCoroutine)
resumeLoadDecks()
end
-- perform the deck importing (with a pause after each deck load)
-- this pause will for example allow weaknesses to be spawned so that the RBW drawing can detect them
function loadDecksCoroutine()
if not allCardsBagApi.isIndexReady() then return end if not allCardsBagApi.isIndexReady() then return end
matsWithInvestigator = playermatApi.getUsedMatColors() matsWithInvestigator = playermatApi.getUsedMatColors()
if redDeckId ~= nil and redDeckId ~= "" then
buildDeck("Red", redDeckId) for _, matColor in ipairs({"White", "Orange", "Green", "Red"}) do
local deckId = _G[string.lower(matColor) .. "DeckId"]
if deckId ~= nil and deckId ~= "" then
buildDeck(matColor, deckId)
coroutine.yield()
end end
if orangeDeckId ~= nil and orangeDeckId ~= "" then
buildDeck("Orange", orangeDeckId)
end end
if whiteDeckId ~= nil and whiteDeckId ~= "" then
buildDeck("White", whiteDeckId)
end end
if greenDeckId ~= nil and greenDeckId ~= "" then
buildDeck("Green", greenDeckId) -- resume the deck importing process
function resumeLoadDecks()
if co and coroutine.status(co) ~= "dead" then
local status, err = coroutine.resume(co)
if not status then error(err) end
end end
end end
@ -247,7 +261,7 @@ function buildDeck(playerColor, deckId)
deckId, deckId,
uiState.privateDeck, uiState.privateDeck,
uiState.loadNewest, uiState.loadNewest,
uiState.investigators, uiState.standalone,
loadCards) loadCards)
end end
@ -343,6 +357,7 @@ function loadCards(slots, investigatorId, bondedList, customizations, playerColo
if (not hadError) then if (not hadError) then
printToAll("Deck loaded successfully!", playerColor) printToAll("Deck loaded successfully!", playerColor)
end end
resumeLoadDecks()
return 1 return 1
end end

View File

@ -40,7 +40,7 @@ local bagSearchers = {}
local hideTitleSplashWaitFunctionId = nil local hideTitleSplashWaitFunctionId = nil
-- online functionality related variables -- online functionality related variables
local MOD_VERSION = "3.9.0" local MOD_VERSION = "3.9.1"
local SOURCE_REPO = 'https://raw.githubusercontent.com/chr1z93/loadable-objects/main' local SOURCE_REPO = 'https://raw.githubusercontent.com/chr1z93/loadable-objects/main'
local library, requestObj, modMeta local library, requestObj, modMeta
local acknowledgedUpgradeVersions = {} local acknowledgedUpgradeVersions = {}
@ -258,7 +258,7 @@ function onObjectNumberTyped(hoveredObject, playerColor, number)
end end
-- check whether the hovered object is part of a players draw objects -- check whether the hovered object is part of a players draw objects
for _, color in ipairs(playermatApi.getUsedMatColors()) do for color, _ in pairs(guidReferenceApi.getObjectsByType("Playermat")) do
local deckAreaObjects = playermatApi.getDeckAreaObjects(color) local deckAreaObjects = playermatApi.getDeckAreaObjects(color)
if deckAreaObjects.topCard == hoveredObject or deckAreaObjects.draw == hoveredObject then if deckAreaObjects.topCard == hoveredObject or deckAreaObjects.draw == hoveredObject then
playermatApi.drawCardsWithReshuffle(color, number) playermatApi.drawCardsWithReshuffle(color, number)

View File

@ -370,19 +370,45 @@ end
-- Gets a random basic weakness from the bag. Once a given ID has been returned it will be -- Gets a random basic weakness from the bag. Once a given ID has been returned it will be
-- removed from the list and cannot be selected again until a reload occurs or the indexes -- removed from the list and cannot be selected again until a reload occurs or the indexes
-- are rebuilt, which will refresh the list to include all weaknesses. -- are rebuilt, which will refresh the list to include all weaknesses.
---@return string: ID of the selected weakness ---@param params table Bundled parameters:
function getRandomWeaknessId() --- count number Number of weaknesses
local availableWeaknesses = buildAvailableWeaknesses() --- restrictions table Additional restrictions:
if #availableWeaknesses > 0 then --- class string Class to restrict weakness to
return availableWeaknesses[math.random(#availableWeaknesses)] --- standalone boolean Whether 'Campaign only' weaknesses should be exluded
--- traits? string Trait(s) to use as filter
---@return table: Table with IDs of the selected weaknesses
function getRandomWeaknessIds(params)
params.count = params.count or 1
local availableWeaknesses = buildAvailableWeaknesses(params.restrictions)
-- check if enough weaknesses are available
local missingWeaknesses = params.count - #availableWeaknesses
if missingWeaknesses > 0 then
broadcastToAll("Not enough basic weaknesses available! (" .. missingWeaknesses .. " missing)", { 0.9, 0.2, 0.2 })
end end
local drawnWeaknesses = {}
-- Fisher-Yates shuffle algorithm
local n = #availableWeaknesses
for i = 1, math.min(params.count, n) do
local index = math.random(i, n)
table.insert(drawnWeaknesses, availableWeaknesses[index])
availableWeaknesses[index], availableWeaknesses[i] = availableWeaknesses[i], availableWeaknesses[index]
end
return drawnWeaknesses
end end
-- Constructs a list of available basic weaknesses by starting with the full pool of basic -- Constructs a list of available basic weaknesses by starting with the full pool of basic
-- weaknesses then removing any which are currently in the play or deck construction areas -- weaknesses then removing any which are currently in the play or deck construction areas
---@param traits? string Trait(s) to use as filter ---@param restrictions? table Additional restrictions:
--- class string Class to restrict weakness to
--- standalone boolean Whether 'Campaign only' weaknesses should be exluded
--- traits? string Trait(s) to use as filter
---@return table: Array of weakness IDs which are valid to choose from ---@return table: Array of weakness IDs which are valid to choose from
function buildAvailableWeaknesses(traits) function buildAvailableWeaknesses(restrictions)
restrictions = restrictions or {}
local weaknessesInPlay = {} local weaknessesInPlay = {}
local allObjects = getAllObjects() local allObjects = getAllObjects()
for _, object in ipairs(allObjects) do for _, object in ipairs(allObjects) do
@ -400,10 +426,29 @@ function buildAvailableWeaknesses(traits)
if (weaknessesInPlay[weaknessId] ~= nil and weaknessesInPlay[weaknessId] > 0) then if (weaknessesInPlay[weaknessId] ~= nil and weaknessesInPlay[weaknessId] > 0) then
weaknessesInPlay[weaknessId] = weaknessesInPlay[weaknessId] - 1 weaknessesInPlay[weaknessId] = weaknessesInPlay[weaknessId] - 1
else else
if traits then local eligible = true
-- disable 'Campaign only' weaknesses in standalone mode
if restrictions.standalone then
local card = cardIdIndex[weaknessId]
if card.metadata.modeRestriction == "Campaign" then
eligible = false
end
end
-- disable class restricted weaknesses
if restrictions.class then
local card = cardIdIndex[weaknessId]
if card.metadata.classRestriction and card.metadata.classRestriction ~= restrictions.class then
eligible = false
end
end
-- disable non-matching traits
if restrictions.traits then
-- split the string into separate traits (separated by "|") -- split the string into separate traits (separated by "|")
local allowedTraits = {} local allowedTraits = {}
for str in traits:gmatch("([^|]+)") do for str in restrictions.traits:gmatch("([^|]+)") do
-- remove dots -- remove dots
str = str:gsub("[%.]", "") str = str:gsub("[%.]", "")
@ -415,15 +460,24 @@ function buildAvailableWeaknesses(traits)
table.insert(allowedTraits, str) table.insert(allowedTraits, str)
end end
local match = false
-- make sure the trait is present on the weakness -- make sure the trait is present on the weakness
local card = cardIdIndex[weaknessId] local card = cardIdIndex[weaknessId]
for _, allowedTrait in ipairs(allowedTraits) do for _, allowedTrait in ipairs(allowedTraits) do
if string.contains(string.lower(card.metadata.traits), allowedTrait) then if string.contains(string.lower(card.metadata.traits), allowedTrait) then
table.insert(availableWeaknesses, weaknessId) match = true
break break
end end
end end
else
if not match then
eligible = false
end
end
-- add weakness to list if eligible
if eligible then
table.insert(availableWeaknesses, weaknessId) table.insert(availableWeaknesses, weaknessId)
end end
end end

View File

@ -28,9 +28,14 @@ do
-- Gets a random basic weakness from the bag. Once a given ID has been returned it -- Gets a random basic weakness from the bag. Once a given ID has been returned it
-- will be removed from the list and cannot be selected again until a reload occurs -- will be removed from the list and cannot be selected again until a reload occurs
-- or the indexes are rebuilt, which will refresh the list to include all weaknesses. -- or the indexes are rebuilt, which will refresh the list to include all weaknesses.
---@return string: ID of the selected weakness ---@param count number Number of weaknesses
AllCardsBagApi.getRandomWeaknessId = function() ---@param restrictions table Additional restrictions:
return getAllCardsBag().call("getRandomWeaknessId") --- class string Class to restrict weakness to
--- standalone boolean Whether 'Campaign only' weaknesses should be exluded
--- traits? string Trait(s) to use as filter
---@return table: Table with IDs of the selected weaknesses
AllCardsBagApi.getRandomWeaknessIds = function(count, restrictions)
return returnCopyOfList(getAllCardsBag().call("getRandomWeaknessIds", {count = count, restrictions = restrictions}))
end end
AllCardsBagApi.isIndexReady = function() AllCardsBagApi.isIndexReady = function()
@ -82,10 +87,13 @@ do
-- Constructs a list of available basic weaknesses by starting with the full pool of basic -- Constructs a list of available basic weaknesses by starting with the full pool of basic
-- weaknesses then removing any which are currently in the play or deck construction areas -- weaknesses then removing any which are currently in the play or deck construction areas
---@param traits? string Trait(s) to use as filter ---@param restrictions table Additional restrictions:
--- class string Class to restrict weakness to
--- standalone boolean Whether 'Campaign only' weaknesses should be exluded
--- traits? string Trait(s) to use as filter
---@return table: Array of weakness IDs which are valid to choose from ---@return table: Array of weakness IDs which are valid to choose from
AllCardsBagApi.buildAvailableWeaknesses = function(traits) AllCardsBagApi.buildAvailableWeaknesses = function(restrictions)
return returnCopyOfList(getAllCardsBag().call("buildAvailableWeaknesses", traits)) return returnCopyOfList(getAllCardsBag().call("buildAvailableWeaknesses", restrictions))
end end
AllCardsBagApi.getUniqueWeaknesses = function() AllCardsBagApi.getUniqueWeaknesses = function()

View File

@ -104,7 +104,9 @@ local tokenColor = {
[""] = "#77674DE6" [""] = "#77674DE6"
} }
function onSave() return JSON.encode(sealedTokens) end function updateSave()
self.script_state = JSON.encode(sealedTokens)
end
function onLoad(savedData) function onLoad(savedData)
sealedTokens = JSON.decode(savedData) or {} sealedTokens = JSON.decode(savedData) or {}
@ -185,6 +187,7 @@ end
function resetSealedTokens() function resetSealedTokens()
sealedTokens = {} sealedTokens = {}
updateSave()
end end
-- native event from TTS - used to update the context menu for cards like "Unrelenting" -- native event from TTS - used to update the context menu for cards like "Unrelenting"
@ -228,6 +231,7 @@ function sealToken(name, playerColor)
end end
end end
updateStackSize() updateStackSize()
updateSave()
end end
}) })
return return
@ -244,6 +248,7 @@ function releaseOneToken(playerColor)
else else
printToColor("Releasing token", playerColor) printToColor("Releasing token", playerColor)
putTokenAway(table.remove(sealedTokens)) putTokenAway(table.remove(sealedTokens))
updateSave()
end end
end end
@ -262,6 +267,7 @@ function releaseMultipleTokens(playerColor)
for i = 1, numRemoved do for i = 1, numRemoved do
putTokenAway(table.remove(sealedTokens)) putTokenAway(table.remove(sealedTokens))
end end
updateSave()
printToColor("Releasing " .. numRemoved .. " tokens", playerColor) printToColor("Releasing " .. numRemoved .. " tokens", playerColor)
end end
@ -276,6 +282,7 @@ function releaseAllTokens(playerColor)
putTokenAway(guid) putTokenAway(guid)
end end
sealedTokens = {} sealedTokens = {}
updateSave()
end end
end end
@ -285,6 +292,7 @@ function returnMultipleTokens(playerColor)
for i = 1, SHOW_MULTI_RETURN do for i = 1, SHOW_MULTI_RETURN do
returnToken(table.remove(sealedTokens)) returnToken(table.remove(sealedTokens))
end end
updateSave()
printToColor("Returning " .. SHOW_MULTI_RETURN .. " tokens", playerColor) printToColor("Returning " .. SHOW_MULTI_RETURN .. " tokens", playerColor)
else else
printToColor("Not enough tokens sealed.", playerColor) printToColor("Not enough tokens sealed.", playerColor)
@ -331,6 +339,7 @@ function resolveSealed()
local resolvedToken = getObjectFromGUID(guidToBeResolved) local resolvedToken = getObjectFromGUID(guidToBeResolved)
resolvedToken.UI.setXml("") resolvedToken.UI.setXml("")
updateStackSize() updateStackSize()
updateSave()
chaosBagApi.drawChaosToken(mat, true, _, guidToBeResolved) chaosBagApi.drawChaosToken(mat, true, _, guidToBeResolved)
end end

View File

@ -603,9 +603,14 @@ function spawnStarterDeck(investigatorName, investigatorData, position)
local cardIdList = {} local cardIdList = {}
for id, count in pairs(slots) do for id, count in pairs(slots) do
for i = 1, count do for i = 1, count do
-- don't include mini-cards and investigators
local card = allCardsBagApi.getCardById(id)
if card and card.metadata and card.metadata.type ~= "Investigator" and card.metadata.type ~= "Minicard" then
table.insert(cardIdList, id) table.insert(cardIdList, id)
end end
end end
end
spawnBag.spawn({ spawnBag.spawn({
name = investigatorName .. "starter", name = investigatorName .. "starter",
cards = cardIdList, cards = cardIdList,
@ -782,21 +787,17 @@ function spawnRandomWeakness(_, playerColor, isRightClick)
prepareToPlaceCards() prepareToPlaceCards()
if not isRightClick then if not isRightClick then
local weaknessId = allCardsBagApi.getRandomWeaknessId() local weaknessIds = allCardsBagApi.getRandomWeaknessIds(1)
if weaknessId == nil then if weaknessIds[1] then
broadcastToAll("All basic weaknesses are in play!", { 0.9, 0.2, 0.2 }) spawnSingleWeakness(weaknessIds[1])
else
spawnSingleWeakness(weaknessId)
end end
else else
Player[playerColor].showInputDialog("Specify a trait for the weakness (split multiple eligible traits with '|'):", lastWeaknessTrait, Player[playerColor].showInputDialog("Specify a trait for the weakness (split multiple eligible traits with '|'):", lastWeaknessTrait,
function(text) function(text)
lastWeaknessTrait = text lastWeaknessTrait = text
local availableWeaknesses = allCardsBagApi.buildAvailableWeaknesses(text) local weaknessIds = allCardsBagApi.getRandomWeaknessIds(1, { traits = text })
if #availableWeaknesses > 0 then if weaknessIds[1] then
spawnSingleWeakness(availableWeaknesses[math.random(#availableWeaknesses)]) spawnSingleWeakness(weaknessIds[1])
else
broadcastToAll("No matching weakness available!", { 0.9, 0.2, 0.2 })
end end
end) end)
end end
@ -808,6 +809,6 @@ function spawnSingleWeakness(weaknessId)
name = "randomWeakness", name = "randomWeakness",
cards = { weaknessId }, cards = { weaknessId },
globalPos = self.positionToWorld(startPositions.randomWeakness), globalPos = self.positionToWorld(startPositions.randomWeakness),
rotation = FACE_UP_ROTATION, rotation = FACE_UP_ROTATION
}) })
end end

View File

@ -261,9 +261,9 @@ INVESTIGATORS["Rex Murphy"] = {
starterDeck = "2624958" starterDeck = "2624958"
} }
INVESTIGATORS["Jenny Barnes"] = { INVESTIGATORS["Jenny Barnes"] = {
cards = { "02003" }, cards = { "02003", "02003-p", "02003-pf", "02003-pb" },
minicards = { "02003-m" }, minicards = { "02003-m" },
signatures = { "02010", "02011", "98002", "98003" }, signatures = { "02010", "02011", "90085", "90086", "98002", "98003" },
starterDeck = "2624961" starterDeck = "2624961"
} }
INVESTIGATORS["Jim Culver"] = { INVESTIGATORS["Jim Culver"] = {

View File

@ -1,4 +1,9 @@
require("playercards/CardsWithHelper") require("playercards/CardsWithHelper")
-- intentionally global
hasXML = false
isHelperEnabled = false
local chaosBagApi = require("chaosbag/ChaosBagApi") local chaosBagApi = require("chaosbag/ChaosBagApi")
-- XML background color for each token -- XML background color for each token
@ -14,18 +19,25 @@ local tokenColor = {
[""] = "#77674DE6" [""] = "#77674DE6"
} }
function onSave() function updateSave()
return JSON.encode(sigil) self.script_state = JSON.encode({
isHelperEnabled = isHelperEnabled,
sigil = sigil
})
end end
function onLoad(savedData) function onLoad(savedData)
self.addContextMenuItem("Enable Helper", chooseSigil) if savedData and savedData ~= "" then
sigil = JSON.decode(savedData) local loadedData = JSON.decode(savedData)
isHelperEnabled = loadedData.isHelperEnabled
sigil = loadedData.sigil
if sigil and sigil ~= nil then if sigil and sigil ~= nil then
makeXMLButton() makeXMLButton()
self.clearContextMenu()
self.addContextMenuItem("Clear Helper", deleteButtons)
end end
self.clearContextMenu()
self.addContextMenuItem("Clear Helper", toggleHelper)
end
syncDisplayWithOptionPanel()
end end
function makeXMLButton() function makeXMLButton()
@ -57,13 +69,42 @@ function makeXMLButton()
} }
} }
) )
updateSave()
end
-- Create XML button to prompt choosing a sigil; acts as this card's helper
function initialize()
if sigil and sigil ~= nil then
makeXMLButton()
else
self.UI.setXmlTable({
{
tag = "Button",
attributes = {
height = 450,
width = 1400,
rotation = "0 0 180",
scale = "0.1 0.1 1",
position = "0 -55 -22",
padding = "50 50 50 50",
font = "font_teutonic-arkham",
fontSize = 300,
onClick = "chooseSigil",
color = "#77674DE6",
textColor = "White"
},
value = "Choose Sigil"
}
}
)
end
end end
-- Create dialog window to choose sigil and create sigil-drawing button -- Create dialog window to choose sigil and create sigil-drawing button
function chooseSigil(playerColor) function chooseSigil(player)
Player[playerColor].clearSelectedObjects() player.clearSelectedObjects()
self.clearContextMenu() self.clearContextMenu()
self.addContextMenuItem("Clear Helper", deleteButtons) self.addContextMenuItem("Clear Helper", toggleHelper)
-- get list of readable names -- get list of readable names
local readableNames = {} local readableNames = {}
@ -72,21 +113,19 @@ function chooseSigil(playerColor)
end end
-- prompt player to choose sigil -- prompt player to choose sigil
Player[playerColor].showOptionsDialog("Choose Sigil", readableNames, 1, player.showOptionsDialog("Choose Sigil", readableNames, 1,
function(chosenToken) function(chosenToken)
sigil = Global.call("getChaosTokenName", chosenToken) sigil = Global.call("getChaosTokenName", chosenToken)
makeXMLButton() makeXMLButton()
end end
) )
updateSave()
end end
-- Delete button and remove sigil function shutOff()
function deleteButtons(playerColor)
Player[playerColor].clearSelectedObjects()
self.clearContextMenu()
self.addContextMenuItem("Enable Helper", chooseSigil)
self.UI.setXml("") self.UI.setXml("")
sigil = nil sigil = nil
updateSave()
end end
function resolveSigil() function resolveSigil()

View File

@ -451,14 +451,12 @@ function drawCardsWithReshuffle(numCards)
local deckAreaObjects = getDeckAreaObjects() local deckAreaObjects = getDeckAreaObjects()
-- Norman Withers handling -- Norman Withers handling
local harbinger = false local harbinger
if deckAreaObjects.topCard and deckAreaObjects.topCard.getName() == "The Harbinger" then if deckAreaObjects.topCard then
harbinger = true harbinger = isHarbinger(deckAreaObjects.topCard.getGMNotes())
elseif deckAreaObjects.draw and not deckAreaObjects.draw.is_face_down then elseif deckAreaObjects.draw and not deckAreaObjects.draw.is_face_down then
local cards = deckAreaObjects.draw.getObjects() local cards = deckAreaObjects.draw.getObjects()
if cards[#cards].name == "The Harbinger" then harbinger = isHarbinger(cards[#cards].gm_notes)
harbinger = true
end
end end
if harbinger then if harbinger then
@ -486,8 +484,7 @@ function drawCardsWithReshuffle(numCards)
if deckSize >= numCards then if deckSize >= numCards then
drawCards(numCards) drawCards(numCards)
-- flip top card again for Norman if topCardDetected then
if topCardDetected and string.match(activeInvestigatorId, "%d%d%d%d%d") == "08004" then
flipTopCardFromDeck() flipTopCardFromDeck()
end end
else else
@ -496,8 +493,7 @@ function drawCardsWithReshuffle(numCards)
shuffleDiscardIntoDeck() shuffleDiscardIntoDeck()
Wait.time(function() Wait.time(function()
drawCards(numCards - deckSize) drawCards(numCards - deckSize)
-- flip top card again for Norman if topCardDetected then
if topCardDetected and string.match(activeInvestigatorId, "%d%d%d%d%d") == "08004" then
flipTopCardFromDeck() flipTopCardFromDeck()
end end
end, 1) end, 1)
@ -506,6 +502,11 @@ function drawCardsWithReshuffle(numCards)
end end
end end
function isHarbinger(notes)
local md = JSON.decode(notes or "") or {}
return md.id == "08006"
end
-- get the draw deck and discard pile objects and returns the references -- get the draw deck and discard pile objects and returns the references
---@return table: string-indexed table with references to the found objects ---@return table: string-indexed table with references to the found objects
function getDeckAreaObjects() function getDeckAreaObjects()