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
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
|
package at.hannibal2.skyhanni.data
import at.hannibal2.skyhanni.SkyHanniMod
import at.hannibal2.skyhanni.config.ConfigUpdaterMigrator
import at.hannibal2.skyhanni.data.FameRanks.getFameRankByNameOrNull
import at.hannibal2.skyhanni.events.BitsUpdateEvent
import at.hannibal2.skyhanni.events.InventoryFullyOpenedEvent
import at.hannibal2.skyhanni.events.LorenzChatEvent
import at.hannibal2.skyhanni.events.ScoreboardChangeEvent
import at.hannibal2.skyhanni.features.misc.NoBitsWarning.sendBitsGainChatMessage
import at.hannibal2.skyhanni.test.command.ErrorManager
import at.hannibal2.skyhanni.utils.CollectionUtils.nextAfter
import at.hannibal2.skyhanni.utils.ItemUtils.getLore
import at.hannibal2.skyhanni.utils.LorenzUtils
import at.hannibal2.skyhanni.utils.NumberUtil.formatInt
import at.hannibal2.skyhanni.utils.SimpleTimeMark
import at.hannibal2.skyhanni.utils.SimpleTimeMark.Companion.asTimeMark
import at.hannibal2.skyhanni.utils.StringUtils.matchFirst
import at.hannibal2.skyhanni.utils.StringUtils.matchMatcher
import at.hannibal2.skyhanni.utils.StringUtils.matches
import at.hannibal2.skyhanni.utils.StringUtils.removeResets
import at.hannibal2.skyhanni.utils.StringUtils.trimWhiteSpace
import at.hannibal2.skyhanni.utils.TimeUtils
import at.hannibal2.skyhanni.utils.repopatterns.RepoPattern
import net.minecraftforge.fml.common.eventhandler.SubscribeEvent
import kotlin.time.Duration.Companion.days
object BitsAPI {
private val profileStorage get() = ProfileStorageData.profileSpecific?.bits
private val playerStorage get() = SkyHanniMod.feature.storage
var bits: Int
get() = profileStorage?.bits ?: 0
private set(value) {
profileStorage?.bits = value
}
var currentFameRank: FameRank?
get() = playerStorage?.currentFameRank?.let { getFameRankByNameOrNull(it) }
private set(value) {
if (value != null) {
playerStorage?.currentFameRank = value.name
}
}
var bitsAvailable: Int
get() = profileStorage?.bitsAvailable ?: 0
private set(value) {
profileStorage?.bitsAvailable = value
}
var cookieBuffTime: SimpleTimeMark?
get() = profileStorage?.boosterCookieExpiryTime?.asTimeMark()
private set(value) {
profileStorage?.boosterCookieExpiryTime = value?.toMillis()
}
private const val defaultcookiebits = 4800
private val bitsDataGroup = RepoPattern.group("data.bits")
// Scoreboard patterns
val bitsScoreboardPattern by bitsDataGroup.pattern(
"scoreboard",
"^Bits: §b(?<amount>[\\d,.]+).*$"
)
// Chat patterns
private val bitsChatGroup = bitsDataGroup.group("chat")
private val bitsFromFameRankUpChatPattern by bitsChatGroup.pattern(
"rankup.bits",
"§eYou gained §3(?<amount>.*) Bits Available §ecompounded from all your §epreviously eaten §6cookies§e! Click here to open §6cookie menu§e!"
)
private val fameRankUpPattern by bitsChatGroup.pattern(
"rankup.rank",
"[§\\w\\s]+FAME RANK UP (?:§.)+(?<rank>.*)"
)
private val boosterCookieAte by bitsChatGroup.pattern(
"boostercookieate",
"§eYou consumed a §6Booster Cookie§e!.*"
)
// GUI patterns
private val bitsGuiGroup = bitsDataGroup.group("gui")
private val bitsAvailableMenuPattern by bitsGuiGroup.pattern(
"availablemenu",
"§7Bits Available: §b(?<toClaim>[\\d,]+)(§3.+)?"
)
private val fameRankSbMenuPattern by bitsGuiGroup.pattern(
"sbmenufamerank",
"§7Your rank: §e(?<rank>.*)"
)
/**
* REGEX-TEST: §7Duration: §a140d 8h 35m 36s
*/
private val cookieDurationPattern by bitsGuiGroup.pattern(
"cookieduration",
"\\s*§7Duration: §a(?<time>.*)"
)
private val noCookieActiveSBMenuPattern by bitsGuiGroup.pattern(
"sbmenunocookieactive",
" §7Status: §cNot active!"
)
private val noCookieActiveCookieMenuPattern by bitsGuiGroup.pattern(
"cookiemenucookieactive",
"(§7§cYou do not currently have a|§cBooster Cookie active!)"
)
private val fameRankCommunityShopPattern by bitsGuiGroup.pattern(
"communityshopfamerank",
"§7Fame Rank: §e(?<rank>.*)"
)
private val bitsGuiNamePattern by bitsGuiGroup.pattern(
"mainmenuname",
"^SkyBlock Menu$"
)
private val cookieGuiStackPattern by bitsGuiGroup.pattern(
"mainmenustack",
"^§6Booster Cookie$"
)
private val bitsStackPattern by bitsGuiGroup.pattern(
"bitsstack",
"§bBits"
)
private val fameRankGuiNamePattern by bitsGuiGroup.pattern(
"famerankmenuname",
"^(Community Shop|Booster Cookie)$"
)
private val fameRankGuiStackPattern by bitsGuiGroup.pattern(
"famerankmenustack",
"^(§aCommunity Shop|§eFame Rank)$"
)
@SubscribeEvent
fun onScoreboardChange(event: ScoreboardChangeEvent) {
if (!isEnabled()) return
for (line in event.newList) {
val message = line.trimWhiteSpace().removeResets()
bitsScoreboardPattern.matchMatcher(message) {
val amount = group("amount").formatInt()
if (amount == bits) return
if (amount > bits) {
bitsAvailable -= amount - bits
sendBitsGainChatMessage(amount - bits)
bits = amount
sendBitsGainEvent()
} else {
bits = amount
sendBitsSpentEvent()
}
}
}
}
@SubscribeEvent
fun onChat(event: LorenzChatEvent) {
if (!isEnabled()) return
val message = event.message.trimWhiteSpace().removeResets()
bitsFromFameRankUpChatPattern.matchMatcher(message) {
val amount = group("amount").formatInt()
bitsAvailable += amount
sendBitsAvailableGainedEvent()
return
}
fameRankUpPattern.matchMatcher(message) {
val rank = group("rank")
currentFameRank = getFameRankByNameOrNull(rank)
?: return ErrorManager.logErrorWithData(
FameRankNotFoundException(rank),
"FameRank $rank not found",
"Rank" to rank,
"Message" to message,
"FameRanks" to FameRanks.fameRanks
)
return
}
boosterCookieAte.matchMatcher(message) {
bitsAvailable += (defaultcookiebits * (currentFameRank?.bitsMultiplier ?: return)).toInt()
val cookieTime = cookieBuffTime
cookieBuffTime = if (cookieTime == null) SimpleTimeMark.now() + 4.days else cookieTime + 4.days
sendBitsAvailableGainedEvent()
return
}
}
@SubscribeEvent
fun onInventoryOpen(event: InventoryFullyOpenedEvent) {
if (!isEnabled()) return
val stacks = event.inventoryItems
if (bitsGuiNamePattern.matches(event.inventoryName)) {
val cookieStack = stacks.values.lastOrNull { cookieGuiStackPattern.matches(it.displayName) }
// If the cookie stack is null, then the player should not have any bits to claim
if (cookieStack == null) {
bitsAvailable = 0
cookieBuffTime = SimpleTimeMark.farPast()
return
}
val lore = cookieStack.getLore()
lore.matchFirst(bitsAvailableMenuPattern) {
val amount = group("toClaim").formatInt()
if (bitsAvailable != amount) {
bitsAvailable = amount
sendBitsAvailableGainedEvent()
val difference = bits - bitsAvailable
if (difference > 0) {
sendBitsGainChatMessage(difference)
bits += difference
}
}
}
lore.matchFirst(cookieDurationPattern) {
val duration = TimeUtils.getDuration(group("time"))
cookieBuffTime = SimpleTimeMark.now() + duration
}
lore.matchFirst(noCookieActiveSBMenuPattern) {
val cookieTime = cookieBuffTime
if (cookieTime == null || cookieTime.isInFuture()) cookieBuffTime = SimpleTimeMark.farPast()
}
return
}
if (fameRankGuiNamePattern.matches(event.inventoryName)) {
val bitsStack = stacks.values.lastOrNull { bitsStackPattern.matches(it.displayName) } ?: return
val fameRankStack = stacks.values.lastOrNull { fameRankGuiStackPattern.matches(it.displayName) } ?: return
val cookieStack = stacks.values.lastOrNull { cookieGuiStackPattern.matches(it.displayName) } ?: return
line@ for (line in fameRankStack.getLore()) {
fameRankCommunityShopPattern.matchMatcher(line) {
val rank = group("rank")
currentFameRank = getFameRankByNameOrNull(rank)
?: return ErrorManager.logErrorWithData(
FameRankNotFoundException(rank),
"FameRank $rank not found",
"Rank" to rank,
"Lore" to fameRankStack.getLore(),
"FameRanks" to FameRanks.fameRanks
)
continue@line
}
fameRankSbMenuPattern.matchMatcher(line) {
val rank = group("rank")
currentFameRank = getFameRankByNameOrNull(rank)
?: return ErrorManager.logErrorWithData(
FameRankNotFoundException(rank),
"FameRank $rank not found",
"Rank" to rank,
"Lore" to fameRankStack.getLore(),
"FameRanks" to FameRanks.fameRanks
)
continue@line
}
}
line@ for (line in bitsStack.getLore()) {
bitsAvailableMenuPattern.matchMatcher(line) {
val amount = group("toClaim").formatInt()
if (amount != bitsAvailable) {
bitsAvailable = amount
sendBitsAvailableGainedEvent()
}
continue@line
}
}
line@ for (line in cookieStack.getLore()) {
cookieDurationPattern.matchMatcher(line) {
val duration = TimeUtils.getDuration(group("time"))
cookieBuffTime = SimpleTimeMark.now().plus(duration)
}
if (noCookieActiveCookieMenuPattern.matches(line)) {
val nextLine = cookieStack.getLore().nextAfter(line) ?: continue@line
if (noCookieActiveCookieMenuPattern.matches(nextLine)) cookieBuffTime = SimpleTimeMark.farPast()
}
}
}
}
fun hasCookieBuff() = cookieBuffTime?.isInFuture() ?: false
private fun sendBitsGainEvent() = BitsUpdateEvent.BitsGain(bits, bitsAvailable).postAndCatch()
private fun sendBitsSpentEvent() = BitsUpdateEvent.BitsSpent(bits, bitsAvailable).postAndCatch()
private fun sendBitsAvailableGainedEvent() = BitsUpdateEvent.BitsAvailableGained(bits, bitsAvailable).postAndCatch()
fun isEnabled() = LorenzUtils.inSkyBlock && profileStorage != null
@SubscribeEvent
fun onConfigFix(event: ConfigUpdaterMigrator.ConfigFixEvent) {
event.move(35, "#profile.bits.bitsToClaim", "#profile.bits.bitsAvailable")
}
class FameRankNotFoundException(rank: String) : Exception("FameRank not found: $rank")
}
|