From f46626df63634168dec63044f561d8da51a5df20 Mon Sep 17 00:00:00 2001
From: My-Name-Is-Jeff <37018278+My-Name-Is-Jeff@users.noreply.github.com>
Date: Fri, 13 Dec 2024 12:11:02 -0500
Subject: [PATCH] feat: Quad Link Legacy solver
---
.../kotlin/gg/skytils/skytilsmod/Skytils.kt | 2 +
.../gg/skytils/skytilsmod/core/Config.kt | 11 +
.../impl/rift/solvers/QuadLinkLegacySolver.kt | 253 ++++++++++++++++++
.../resources/assets/skytils/lang/en_US.lang | 2 +
4 files changed, 268 insertions(+)
create mode 100644 src/main/kotlin/gg/skytils/skytilsmod/features/impl/rift/solvers/QuadLinkLegacySolver.kt
diff --git a/src/main/kotlin/gg/skytils/skytilsmod/Skytils.kt b/src/main/kotlin/gg/skytils/skytilsmod/Skytils.kt
index 63955e336..d00eef443 100644
--- a/src/main/kotlin/gg/skytils/skytilsmod/Skytils.kt
+++ b/src/main/kotlin/gg/skytils/skytilsmod/Skytils.kt
@@ -52,6 +52,7 @@ import gg.skytils.skytilsmod.features.impl.mining.StupidTreasureChestOpeningThin
import gg.skytils.skytilsmod.features.impl.misc.*
import gg.skytils.skytilsmod.features.impl.overlays.AuctionPriceOverlay
import gg.skytils.skytilsmod.features.impl.protectitems.ProtectItems
+import gg.skytils.skytilsmod.features.impl.rift.solvers.QuadLinkLegacySolver
import gg.skytils.skytilsmod.features.impl.slayer.SlayerFeatures
import gg.skytils.skytilsmod.features.impl.spidersden.RainTimer
import gg.skytils.skytilsmod.features.impl.spidersden.RelicWaypoints
@@ -361,6 +362,7 @@ class Skytils {
PotionEffectTimers,
PricePaid,
ProtectItems,
+ QuadLinkLegacySolver,
QuiverStuff,
RainTimer,
RandomStuff,
diff --git a/src/main/kotlin/gg/skytils/skytilsmod/core/Config.kt b/src/main/kotlin/gg/skytils/skytilsmod/core/Config.kt
index f824169ba..2318be329 100644
--- a/src/main/kotlin/gg/skytils/skytilsmod/core/Config.kt
+++ b/src/main/kotlin/gg/skytils/skytilsmod/core/Config.kt
@@ -3418,6 +3418,17 @@ object Config : Vigilant(
)
var petItemConfirmation = false
+ @Property(
+ type = PropertyType.SWITCH, name = "Quad Link Legacy Solver",
+ description = "§b[WIP]§r Solves the Quad Link Legacy (CONNECT4) puzzle.",
+ category = "Rift", subcategory = "Solvers",
+ i18nName = "skytils.config.rift.solvers.quad_link_legacy_solver",
+ i18nCategory = "skytils.config.rift",
+ i18nSubcategory = "skytils.config.rift.solvers",
+ searchTags = ["Wizardman", "Connect4", "ConnectFOUR"],
+ )
+ var quadLinkLegacySolver = false
+
@Property(
type = PropertyType.DECIMAL_SLIDER, name = "Current Revenant RNG Meter",
description = "Internal value to store current Revenant RNG meter",
diff --git a/src/main/kotlin/gg/skytils/skytilsmod/features/impl/rift/solvers/QuadLinkLegacySolver.kt b/src/main/kotlin/gg/skytils/skytilsmod/features/impl/rift/solvers/QuadLinkLegacySolver.kt
new file mode 100644
index 000000000..33d1ed938
--- /dev/null
+++ b/src/main/kotlin/gg/skytils/skytilsmod/features/impl/rift/solvers/QuadLinkLegacySolver.kt
@@ -0,0 +1,253 @@
+/*
+ * Skytils - Hypixel Skyblock Quality of Life Mod
+ * Copyright (C) 2020-2024 Skytils
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as published
+ * by the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Affero General Public License for more details.
+ *
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see .
+ */
+
+package gg.skytils.skytilsmod.features.impl.rift.solvers
+
+import gg.skytils.skytilsmod.Skytils
+import gg.skytils.skytilsmod.Skytils.Companion.mc
+import gg.skytils.skytilsmod.events.impl.GuiContainerEvent
+import gg.skytils.skytilsmod.utils.withAlpha
+import net.minecraft.client.gui.Gui
+import net.minecraft.init.Blocks
+import net.minecraft.init.Items
+import net.minecraft.inventory.ContainerChest
+import net.minecraft.item.Item
+import net.minecraft.item.ItemStack
+import net.minecraftforge.event.world.WorldEvent
+import net.minecraftforge.fml.common.eventhandler.SubscribeEvent
+import java.awt.Color
+
+object QuadLinkLegacySolver {
+ const val guiTitle = "Quad Link Legacy - Wizardman"
+ const val oppSlot = 17
+ const val ourSlot = 24
+
+ // he takes up the middle 7 slots for each row of the chest
+ // this is [row][column]
+ val boardSlots = (0 until 6).map { it*9+1..it*9+7 }
+ val flatBoardSlots = boardSlots.flatten()
+
+ var ourItem: ItemStack? = null
+ var oppItem: ItemStack? = null
+ var bestColumn = -1
+
+ @SubscribeEvent
+ fun onWorldChange(event: WorldEvent.Unload) {
+ if (event.world != mc.theWorld) return
+ reset()
+ }
+
+ @SubscribeEvent
+ fun onGuiContainerEvent(event: GuiContainerEvent) {
+ when (event) {
+ is GuiContainerEvent.CloseWindowEvent -> reset()
+ is GuiContainerEvent.ForegroundDrawnEvent -> {
+ if (!Skytils.config.quadLinkLegacySolver) return
+ val container = event.container as? ContainerChest ?: return
+ if (event.chestName != guiTitle) return
+
+ if (ourItem == null) {
+ ourItem = container.getSlot(ourSlot).stack
+ oppItem = container.getSlot(oppSlot).stack
+
+ check(ourItem != null && oppItem != null) { "Our item or opponent's item is null" }
+ }
+
+ // if null, it means the placing animation is happening
+ // if painting, it means we are waiting for the move
+ // if item stack, it means they are waiting for us
+ // if there is glass, the game is over
+ if (flatBoardSlots.map { container.getSlot(it).stack }.any {
+ it == null ||
+ it.item == Items.painting ||
+ it.item == Item.getItemFromBlock(Blocks.stained_glass) ||
+ !(it.getIsItemStackEqual(ourItem) || it.getIsItemStackEqual(oppItem) || it.item == Items.item_frame)
+ }) {
+ bestColumn = -1
+ return
+ }
+
+ if (bestColumn == -1) {
+ // read the board in
+ for (column in 0 until 7) {
+ for (row in 0 until 6) {
+ val slot = boardSlots[row].elementAt(column)
+ val item = container.getSlot(slot).stack
+ if (item.getIsItemStackEqual(ourItem)) {
+ board[column][row] = true
+ } else if (item.getIsItemStackEqual(oppItem)) {
+ board[column][row] = false
+ } else if (item.item == Items.item_frame) {
+ board[column][row] = null
+ }
+ }
+ }
+
+ val result = negamax(1000, isOurs = true)
+ bestColumn = result.first
+ }
+
+ if (bestColumn != -1) {
+ val topSlot = container.getSlot(bestColumn+1)
+ Gui.drawRect(
+ topSlot.xDisplayPosition,
+ topSlot.yDisplayPosition,
+ topSlot.xDisplayPosition + 16,
+ topSlot.yDisplayPosition + 16 * 6,
+ Color.RED.withAlpha(100)
+ )
+ }
+ }
+ }
+ }
+
+ fun reset() {
+ board.forEach { it.fill(null) }
+ ourItem = null
+ oppItem = null
+ bestColumn = -1
+ }
+
+ /**
+ * board[column][row]
+ * boolean? = null if empty, true if our piece, false if opponent's piece
+ */
+ val board: Array> = Array(7) { arrayOfNulls(6) }
+
+ /**
+ * Makes a move in Connect 4
+ * @return If the move was successful
+ */
+ fun makeMove(column: Int, ourPiece: Boolean): Boolean {
+ check(column in 0 until 7) { "Column must be between 0 and 6" }
+ board[column].forEachIndexed { index, b ->
+ if (b == null) {
+ board[column][index] = ourPiece
+ return true
+ }
+ }
+ return false
+ }
+
+ /**
+ * Removes the top piece on a column in Connect 4
+ * @return If the move was successful
+ */
+ fun popMove(column: Int): Boolean {
+ check(column in 0 until 7) { "Column must be between 0 and 6" }
+ for (row in 5 downTo 0) {
+ if (board[column][row] != null) {
+ board[column][row] = null
+ return true
+ }
+ }
+ return false
+ }
+
+
+ /**
+ * @return true if we won, false if opponent won, null if no one won
+ */
+ fun getWinner(): Boolean? {
+ // Check horizontal
+ for (row in 0 until 6) {
+ for (column in 0 until 4) {
+ if (board[column][row] != null &&
+ board[column][row] == board[column + 1][row] &&
+ board[column][row] == board[column + 2][row] &&
+ board[column][row] == board[column + 3][row]
+ ) {
+ return board[column][row]
+ }
+ }
+ }
+
+ // Check vertical
+ for (column in 0 until 7) {
+ for (row in 0 until 3) {
+ if (board[column][row] != null &&
+ board[column][row] == board[column][row + 1] &&
+ board[column][row] == board[column][row + 2] &&
+ board[column][row] == board[column][row + 3]
+ ) {
+ return board[column][row]
+ }
+ }
+ }
+
+ // Check diagonal
+ for (column in 0 until 4) {
+ for (row in 0 until 3) {
+ if (board[column][row] != null &&
+ board[column][row] == board[column + 1][row + 1] &&
+ board[column][row] == board[column + 2][row + 2] &&
+ board[column][row] == board[column + 3][row + 3]
+ ) {
+ return board[column][row]
+ }
+ }
+ }
+
+ for (column in 0 until 4) {
+ for (row in 3 until 6) {
+ if (board[column][row] != null &&
+ board[column][row] == board[column + 1][row - 1] &&
+ board[column][row] == board[column + 2][row - 2] &&
+ board[column][row] == board[column + 3][row - 3]
+ ) {
+ return board[column][row]
+ }
+ }
+ }
+
+ return null
+ }
+
+ /**
+ * @return The best move to make and the score of that move
+ */
+ fun negamax(depth: Int, alpha: Int = Int.MIN_VALUE, beta: Int = Int.MAX_VALUE, isOurs: Boolean): Pair {
+ // TODO: find better base case score
+ if (depth == 0) return -1 to 0
+ val winner = getWinner()
+ if (winner != null) {
+ return -1 to if (winner) depth * 1000 else -depth * 1000
+ }
+
+ var bestScore = Int.MIN_VALUE
+ var a = alpha
+ var bestMove = -1
+
+ for (column in 0 until 7) {
+ if (makeMove(column, isOurs)) {
+ val score = -negamax(depth - 1, -beta, -a, !isOurs).second
+ popMove(column)
+ if (score >= beta) return column to score
+
+ if (score > bestScore) {
+ bestScore = score
+ bestMove = column
+ }
+ a = maxOf(a, score)
+ if (alpha >= beta) break
+ }
+ }
+
+ return bestMove to bestScore
+ }
+}
\ No newline at end of file
diff --git a/src/main/resources/assets/skytils/lang/en_US.lang b/src/main/resources/assets/skytils/lang/en_US.lang
index 3d44c5148..427894070 100644
--- a/src/main/resources/assets/skytils/lang/en_US.lang
+++ b/src/main/resources/assets/skytils/lang/en_US.lang
@@ -421,6 +421,7 @@ skytils.config.farming=Farming
skytils.config.kuudra=Kuudra
skytils.config.mining=Mining
skytils.config.pets=Pets
+skytils.config.rift=Rift
skytils.config.slayer=Slayer
skytils.config.sounds=Sounds
skytils.config.spam=Spam
@@ -461,6 +462,7 @@ skytils.config.miscellaneous.minions=Minions
skytils.config.miscellaneous.quality_of_life=Quality of Life
skytils.config.miscellaneous.other=Other
skytils.config.pets.quality_of_life=Quality of Life
+skytils.config.rift.solvers=Solvers
skytils.config.slayer.quality_of_life=Quality of Life
skytils.config.slayer.general=General
skytils.config.slayer.voidgloom_seraph=Voidgloom Seraph