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
|
package at.hannibal2.skyhanni.data
import at.hannibal2.skyhanni.data.jsonobjects.repo.GardenJson
import at.hannibal2.skyhanni.events.CropMilestoneUpdateEvent
import at.hannibal2.skyhanni.events.InventoryFullyOpenedEvent
import at.hannibal2.skyhanni.events.RepositoryReloadEvent
import at.hannibal2.skyhanni.features.garden.CropType
import at.hannibal2.skyhanni.features.garden.GardenAPI
import at.hannibal2.skyhanni.skyhannimodule.SkyHanniModule
import at.hannibal2.skyhanni.utils.ChatUtils.chat
import at.hannibal2.skyhanni.utils.ItemUtils.getLore
import at.hannibal2.skyhanni.utils.NumberUtil.formatLong
import at.hannibal2.skyhanni.utils.RegexUtils.matchFirst
import at.hannibal2.skyhanni.utils.SoundUtils
import at.hannibal2.skyhanni.utils.SoundUtils.playSound
import at.hannibal2.skyhanni.utils.repopatterns.RepoPattern
import net.minecraft.item.ItemStack
import net.minecraftforge.fml.common.eventhandler.SubscribeEvent
@SkyHanniModule
object GardenCropMilestones {
private val patternGroup = RepoPattern.group("data.garden.milestone")
private val cropPattern by patternGroup.pattern(
"crop",
"§7Harvest §f(?<name>.*) §7on .*"
)
val totalPattern by patternGroup.pattern(
"total",
"§7Total: §a(?<name>.*)"
)
private val config get() = GardenAPI.config.cropMilestones
fun getCropTypeByLore(itemStack: ItemStack): CropType? {
itemStack.getLore().matchFirst(cropPattern) {
val name = group("name")
return CropType.getByNameOrNull(name)
}
return null
}
@SubscribeEvent
fun onInventoryOpen(event: InventoryFullyOpenedEvent) {
if (event.inventoryName != "Crop Milestones") return
for ((_, stack) in event.inventoryItems) {
val crop = getCropTypeByLore(stack) ?: continue
stack.getLore().matchFirst(totalPattern) {
val amount = group("name").formatLong()
crop.setCounter(amount)
}
}
CropMilestoneUpdateEvent().postAndCatch()
GardenCropMilestonesCommunityFix.openInventory(event.inventoryItems)
}
fun onOverflowLevelUp(crop: CropType, oldLevel: Int, newLevel: Int) {
val customGoalLevel = ProfileStorageData.profileSpecific?.garden?.customGoalMilestone?.get(crop) ?: 0
val goalReached = newLevel == customGoalLevel
// TODO utils function that is shared with Garden Level Display
val rewards = buildList {
add(" §r§8+§aRespect from Elite Farmers and SkyHanni members :)")
add(" §r§8+§b1 Flexing Point")
if (newLevel % 5 == 0)
add(" §r§7§8+§d2 SkyHanni User Luck")
}
val messages = listOf(
"§r§3§l▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬§r",
" §r§b§lGARDEN MILESTONE §3${crop.cropName} §8$oldLevel➜§3$newLevel§r",
if (goalReached)
listOf(
"",
" §r§d§lGOAL REACHED!",
"",
).joinToString("\n")
else
"",
" §r§a§lREWARDS§r",
rewards.joinToString("\n"),
"§r§3§l▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬▬§r"
)
chat(messages.joinToString("\n"), false)
if (goalReached)
chat("§e§lYou have reached your milestone goal of §b§l${customGoalLevel} §e§lin the §b§l${crop.cropName} §e§lcrop!", false)
SoundUtils.createSound("random.levelup", 1f, 1f).playSound()
}
var cropMilestoneData: Map<CropType, List<Int>> = emptyMap()
val cropCounter: MutableMap<CropType, Long>? get() = GardenAPI.storage?.cropCounter
// TODO make nullable
fun CropType.getCounter() = cropCounter?.get(this) ?: 0
fun CropType.setCounter(counter: Long) {
cropCounter?.set(this, counter)
}
fun CropType.isMaxed(useOverflow: Boolean): Boolean {
if (useOverflow) return false
// TODO change 1b
val maxValue = cropMilestoneData[this]?.sum() ?: 1_000_000_000 // 1 bil for now
return getCounter() >= maxValue
}
fun getTierForCropCount(count: Long, crop: CropType, allowOverflow: Boolean = false): Int {
var tier = 0
var totalCrops = 0L
val cropMilestone = cropMilestoneData[crop] ?: return 0
val last = cropMilestone.last()
for (tierCrops in cropMilestone) {
totalCrops += tierCrops
if (totalCrops >= count) {
return tier
}
tier++
}
if (allowOverflow) {
while (totalCrops < count) {
totalCrops += last
if (totalCrops >= count) {
return tier
}
tier++
}
}
return tier
}
fun getMaxTier() = cropMilestoneData.values.firstOrNull()?.size ?: 0
fun getCropsForTier(requestedTier: Int, crop: CropType, allowOverflow: Boolean = false): Long {
var totalCrops = 0L
var tier = 0
val cropMilestone = cropMilestoneData[crop] ?: return 0
val definedTiers = cropMilestone.size
if (requestedTier <= definedTiers || !allowOverflow) {
for (tierCrops in cropMilestone) {
totalCrops += tierCrops
tier++
if (tier == requestedTier) {
return totalCrops
}
}
return if (!allowOverflow) 0 else totalCrops
}
for (tierCrops in cropMilestone) {
totalCrops += tierCrops
tier++
}
val additionalTiers = requestedTier - definedTiers
val lastIncrement = cropMilestone.last().toLong()
totalCrops += lastIncrement * additionalTiers
return totalCrops
}
fun CropType.progressToNextLevel(allowOverflow: Boolean = false): Double {
val progress = getCounter()
val startTier = getTierForCropCount(progress, this, allowOverflow)
val startCrops = getCropsForTier(startTier, this, allowOverflow)
val end = getCropsForTier(startTier + 1, this, allowOverflow)
return (progress - startCrops).toDouble() / (end - startCrops)
}
@SubscribeEvent
fun onRepoReload(event: RepositoryReloadEvent) {
cropMilestoneData = event.getConstant<GardenJson>("Garden").crop_milestones
}
}
|