1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
|
package at.hannibal2.skyhanni.features.slayer
import at.hannibal2.skyhanni.SkyHanniMod
import at.hannibal2.skyhanni.config.ConfigUpdaterMigrator
import at.hannibal2.skyhanni.config.storage.ProfileSpecificStorage
import at.hannibal2.skyhanni.data.SlayerAPI
import at.hannibal2.skyhanni.data.jsonobjects.repo.SlayerProfitTrackerItemsJson
import at.hannibal2.skyhanni.events.GuiRenderEvent
import at.hannibal2.skyhanni.events.ItemAddEvent
import at.hannibal2.skyhanni.events.LorenzChatEvent
import at.hannibal2.skyhanni.events.PurseChangeCause
import at.hannibal2.skyhanni.events.PurseChangeEvent
import at.hannibal2.skyhanni.events.RepositoryReloadEvent
import at.hannibal2.skyhanni.events.SlayerChangeEvent
import at.hannibal2.skyhanni.events.SlayerQuestCompleteEvent
import at.hannibal2.skyhanni.skyhannimodule.SkyHanniModule
import at.hannibal2.skyhanni.utils.ChatUtils
import at.hannibal2.skyhanni.utils.CollectionUtils.addAsSingletonList
import at.hannibal2.skyhanni.utils.LorenzUtils
import at.hannibal2.skyhanni.utils.NEUInternalName
import at.hannibal2.skyhanni.utils.NumberUtil
import at.hannibal2.skyhanni.utils.NumberUtil.addSeparators
import at.hannibal2.skyhanni.utils.NumberUtil.formatDouble
import at.hannibal2.skyhanni.utils.RegexUtils.matchMatcher
import at.hannibal2.skyhanni.utils.StringUtils.removeColor
import at.hannibal2.skyhanni.utils.renderables.Renderable
import at.hannibal2.skyhanni.utils.repopatterns.RepoPattern
import at.hannibal2.skyhanni.utils.tracker.ItemTrackerData
import at.hannibal2.skyhanni.utils.tracker.SkyHanniItemTracker
import com.google.gson.JsonObject
import com.google.gson.JsonPrimitive
import com.google.gson.annotations.Expose
import net.minecraftforge.fml.common.eventhandler.SubscribeEvent
@SkyHanniModule
object SlayerProfitTracker {
private val config get() = SkyHanniMod.feature.slayer.itemProfitTracker
private var itemLogCategory = ""
private var baseSlayerType = ""
private val trackers = mutableMapOf<String, SkyHanniItemTracker<Data>>()
/**
* REGEX-TEST: §7Took 1.9k coins from your bank for auto-slayer...
*/
private val autoSlayerBankPattern by RepoPattern.pattern(
"slayer.autoslayer.bank.chat",
"§7Took (?<coins>.+) coins from your bank for auto-slayer\\.\\.\\."
)
class Data : ItemTrackerData() {
override fun resetItems() {
slayerSpawnCost = 0
slayerCompletedCount = 0
}
@Expose
var slayerSpawnCost: Long = 0
@Expose
var slayerCompletedCount = 0L
override fun getDescription(timesDropped: Long): List<String> {
val percentage = timesDropped.toDouble() / slayerCompletedCount
val perBoss = LorenzUtils.formatPercentage(percentage.coerceAtMost(1.0))
return listOf(
"§7Dropped §e${timesDropped.addSeparators()} §7times.",
"§7Your drop rate: §c$perBoss",
)
}
override fun getCoinName(item: TrackedItem) = "§6Mob Kill Coins"
override fun getCoinDescription(item: TrackedItem): List<String> {
val mobKillCoinsFormat = NumberUtil.format(item.totalAmount)
return listOf(
"§7Killing mobs gives you coins (more with scavenger).",
"§7You got §6$mobKillCoinsFormat coins §7that way."
)
}
}
private val ItemTrackerData.TrackedItem.timesDropped get() = timesGained
private fun addSlayerCosts(price: Double) {
require(price < 0) { "slayer costs can not be positve" }
getTracker()?.modify {
it.slayerSpawnCost += price.toInt()
}
}
private var allowedItems = mapOf<String, List<NEUInternalName>>()
@SubscribeEvent
fun onRepoReload(event: RepositoryReloadEvent) {
allowedItems = event.getConstant<SlayerProfitTrackerItemsJson>("SlayerProfitTrackerItems").slayers
}
@SubscribeEvent
fun onPurseChange(event: PurseChangeEvent) {
if (!isEnabled()) return
val coins = event.coins
if (event.reason == PurseChangeCause.GAIN_MOB_KILL && SlayerAPI.isInCorrectArea) {
getTracker()?.addCoins(coins.toInt())
}
if (event.reason == PurseChangeCause.LOSE_SLAYER_QUEST_STARTED) {
addSlayerCosts(coins)
}
}
@SubscribeEvent
fun onChat(event: LorenzChatEvent) {
if (!isEnabled()) return
autoSlayerBankPattern.matchMatcher(event.message) {
addSlayerCosts(-group("coins").formatDouble())
}
}
@SubscribeEvent
fun onSlayerChange(event: SlayerChangeEvent) {
val newSlayer = event.newSlayer
itemLogCategory = newSlayer.removeColor()
baseSlayerType = itemLogCategory.substringBeforeLast(" ")
getTracker()?.update()
}
private fun getTracker(): SkyHanniItemTracker<Data>? {
if (itemLogCategory == "") return null
return trackers.getOrPut(itemLogCategory) {
val getStorage: (ProfileSpecificStorage) -> Data = {
it.slayerProfitData.getOrPut(
itemLogCategory
) { Data() }
}
SkyHanniItemTracker("$itemLogCategory Profit Tracker", { Data() }, getStorage) { drawDisplay(it) }
}
}
@SubscribeEvent
fun onQuestComplete(event: SlayerQuestCompleteEvent) {
getTracker()?.modify {
it.slayerCompletedCount++
}
}
@SubscribeEvent
fun onItemAdd(event: ItemAddEvent) {
if (!isEnabled()) return
if (!SlayerAPI.isInCorrectArea) return
if (!SlayerAPI.hasActiveSlayerQuest()) return
val internalName = event.internalName
val amount = event.amount
if (!isAllowedItem(internalName)) {
ChatUtils.debug("Ignored non-slayer item pickup: '$internalName' '$itemLogCategory'")
return
}
getTracker()?.addItem(internalName, amount)
}
private fun isAllowedItem(internalName: NEUInternalName): Boolean {
val allowedList = allowedItems[baseSlayerType] ?: return false
return internalName in allowedList
}
private fun drawDisplay(data: Data) = buildList<List<Any>> {
val tracker = getTracker() ?: return@buildList
addAsSingletonList("§e§l$itemLogCategory Profit Tracker")
var profit = tracker.drawItems(data, { true }, this)
val slayerSpawnCost = data.slayerSpawnCost
if (slayerSpawnCost != 0L) {
val mobKillCoinsFormat = NumberUtil.format(slayerSpawnCost)
addAsSingletonList(
Renderable.hoverTips(
" §7Slayer Spawn Costs: §c$mobKillCoinsFormat",
listOf("§7You paid §c$mobKillCoinsFormat §7in total", "§7for starting the slayer quests.")
)
)
profit += slayerSpawnCost
}
val slayerCompletedCount = data.slayerCompletedCount
addAsSingletonList(
Renderable.hoverTips(
"§7Bosses killed: §e${slayerCompletedCount.addSeparators()}",
listOf("§7You killed the $itemLogCategory boss", "§e${slayerCompletedCount.addSeparators()} §7times.")
)
)
addAsSingletonList(tracker.addTotalProfit(profit, data.slayerCompletedCount, "boss"))
tracker.addPriceFromButton(this)
}
val coinFormat: (ItemTrackerData.TrackedItem) -> Pair<String, List<String>> = { item ->
val mobKillCoinsFormat = NumberUtil.format(item.totalAmount)
val text = " §6Mob kill coins§7: §6$mobKillCoinsFormat"
val lore = listOf(
"§7Killing mobs gives you coins (more with scavenger)",
"§7You got §e$mobKillCoinsFormat §7coins in total this way"
)
text to lore
}
@SubscribeEvent
fun onRenderOverlay(event: GuiRenderEvent) {
if (!isEnabled()) return
if (!SlayerAPI.isInCorrectArea) return
getTracker()?.renderDisplay(config.pos)
}
@SubscribeEvent
fun onConfigFix(event: ConfigUpdaterMigrator.ConfigFixEvent) {
event.transform(10, "#profile.slayerProfitData") { old ->
for (data in old.asJsonObject.entrySet().map { it.value.asJsonObject }) {
val items = data.get("items").asJsonObject
for (item in items.entrySet().map { it.value.asJsonObject }) {
val oldValue = item.get("timesDropped")
item.add("timesGained", oldValue)
}
val coinAmount = data.get("mobKillCoins")
val coins = JsonObject()
coins.add("internalName", JsonPrimitive("SKYBLOCK_COIN"))
coins.add("timesDropped", JsonPrimitive(1))
coins.add("totalAmount", coinAmount)
items.add("SKYBLOCK_COIN", coins)
}
old
}
}
fun isEnabled() = LorenzUtils.inSkyBlock && config.enabled
fun clearProfitCommand(args: Array<String>) {
if (itemLogCategory == "")
|