From 0fbe5a18b811e2a1587d7af994e5b61a47e35ba2 Mon Sep 17 00:00:00 2001 From: SeventhM <127357473+SeventhM@users.noreply.github.com> Date: Tue, 8 Aug 2023 23:17:09 -0700 Subject: [PATCH] Fix promotions being unavailable when it can be reached a slower way (#9856) Previously if a promotion could require 2 promotions from branch A but 1 branch B and the branch B also lead to an earlier promotion from branch A, it erroneously consider that a loop. Solved by copying the logic from the ruleset Validator (going child up to parent, rather than parent to child and passing in the hashset rather than using a global one) --- .../ui/screens/pickerscreens/PromotionTree.kt | 18 ++++++++---------- 1 file changed, 8 insertions(+), 10 deletions(-) diff --git a/core/src/com/unciv/ui/screens/pickerscreens/PromotionTree.kt b/core/src/com/unciv/ui/screens/pickerscreens/PromotionTree.kt index a8be3cc55d..d5b5736009 100644 --- a/core/src/com/unciv/ui/screens/pickerscreens/PromotionTree.kt +++ b/core/src/com/unciv/ui/screens/pickerscreens/PromotionTree.kt @@ -89,9 +89,9 @@ internal class PromotionTree(val unit: MapUnit) { // Fill parent/child relations, ignoring prerequisites not in possiblePromotions for (node in nodes.values) { + if (detectLoop(node)) continue for (prerequisite in node.promotion.prerequisites) { val parent = nodes[prerequisite] ?: continue - if (detectLoop(node, parent)) continue node.parents += parent parent.children += node if (node.level > 0 && node.baseName == parent.baseName) @@ -150,20 +150,18 @@ internal class PromotionTree(val unit: MapUnit) { fun allNodes() = nodes.values.asSequence() fun allRoots() = allNodes().filter { it.isRoot } - private fun detectLoop(node: PromotionNode, parent: PromotionNode): Boolean { - if (parent == node) return true + private fun detectLoop(node: PromotionNode): Boolean { val loopCheck = HashSet(nodes.size) - loopCheck.add(node) - fun detectRecursive(parent: PromotionNode, level: Int): Boolean { + fun detectRecursive(node: PromotionNode, level: Int, loopCheck: HashSet): Boolean { if (level > 99) return true - if (parent in loopCheck) return true - loopCheck.add(parent) - for (child in parent.children) { - if (detectRecursive(child, level + 1)) return true + if (node in loopCheck) return true + loopCheck.add(node) + for (parent in node.parents) { + if (detectRecursive(parent, level + 1, loopCheck)) return true } return false } - return detectRecursive(parent, 0) + return detectRecursive(node, 0, loopCheck) } private fun getReachableNode(promotion: Promotion): PromotionNode? =