package emu.grasscutter.game.gacha; import static emu.grasscutter.config.Configuration.*; import emu.grasscutter.Grasscutter; import emu.grasscutter.data.common.ItemParamData; import emu.grasscutter.game.player.Player; import emu.grasscutter.net.proto.GachaInfoOuterClass.GachaInfo; import emu.grasscutter.net.proto.GachaUpInfoOuterClass.GachaUpInfo; import emu.grasscutter.utils.Utils; import lombok.Getter; public class GachaBanner { @Getter private int gachaType; @Getter private int scheduleId; @Getter private String prefabPath; private String previewPrefabPath; @Getter private String titlePath; private int costItemId = 0; private int costItemAmount = 1; private int costItemId10 = 0; private int costItemAmount10 = 10; @Getter private int beginTime; @Getter private int endTime; @Getter private int sortId; @Getter private int gachaTimesLimit = Integer.MAX_VALUE; private int[] rateUpItems4 = {}; private int[] rateUpItems5 = {}; @Getter private int[] fallbackItems3 = {11301, 11302, 11306, 12301, 12302, 12305, 13303, 14301, 14302, 14304, 15301, 15302, 15304}; @Getter private int[] fallbackItems4Pool1 = {1014, 1020, 1023, 1024, 1025, 1027, 1031, 1032, 1034, 1036, 1039, 1043, 1044, 1045, 1048, 1053, 1055, 1056, 1064}; @Getter private int[] fallbackItems4Pool2 = {11401, 11402, 11403, 11405, 12401, 12402, 12403, 12405, 13401, 13407, 14401, 14402, 14403, 14409, 15401, 15402, 15403, 15405}; @Getter private int[] fallbackItems5Pool1 = {1003, 1016, 1042, 1035, 1041}; @Getter private int[] fallbackItems5Pool2 = {11501, 11502, 12501, 12502, 13502, 13505, 14501, 14502, 15501, 15502}; @Getter private boolean removeC6FromPool = false; @Getter private boolean autoStripRateUpFromFallback = true; private int[][] weights4 = {{1,510}, {8,510}, {10,10000}}; private int[][] weights5 = {{1,75}, {73,150}, {90,10000}}; private int[][] poolBalanceWeights4 = {{1,255}, {17,255}, {21,10455}}; private int[][] poolBalanceWeights5 = {{1,30}, {147,150}, {181,10230}}; private int eventChance4 = 50; // Chance to win a featured event item private int eventChance5 = 50; // Chance to win a featured event item @Getter private BannerType bannerType = BannerType.STANDARD; @Getter private int wishMaxProgress = 2; // Deprecated fields that were tolerated in early May 2022 but have apparently still being circulating in new custom configs // For now, throw up big scary errors on load telling people that they will be banned outright in a future version @Deprecated private int[] rateUpItems1 = {}; @Deprecated private int[] rateUpItems2 = {}; @Deprecated private int eventChance = -1; @Deprecated private int costItem = 0; @Deprecated private int softPity = -1; @Deprecated private int hardPity = -1; @Deprecated private int minItemType = -1; @Deprecated private int maxItemType = -1; private static void warnDeprecated(String name, String replacement) { Grasscutter.getLogger().error("Deprecated field found in Banners.json: "+name+" was replaced back in early May 2022, use "+replacement+" instead. If you do not remove this key from your config, it will refuse to load in a future Grasscutter version."); } public void onLoad() { if (eventChance != -1) warnDeprecated("eventChance", "eventChance4 & eventChance5"); if (costItem != 0) warnDeprecated("costItem", "costItemId"); if (softPity != -1) warnDeprecated("softPity", "weights5"); if (hardPity != -1) warnDeprecated("hardPity", "weights5"); if (minItemType != -1) warnDeprecated("minItemType", "fallbackItems[4,5]Pool[1,2]"); if (maxItemType != -1) warnDeprecated("maxItemType", "fallbackItems[4,5]Pool[1,2]"); if (rateUpItems1.length > 0) warnDeprecated("rateUpItems1", "rateUpItems5"); if (rateUpItems2.length > 0) warnDeprecated("rateUpItems2", "rateUpItems4"); } public String getPreviewPrefabPath() { if (this.previewPrefabPath != null && !this.previewPrefabPath.isEmpty()) return this.previewPrefabPath; return "UI_Tab_" + this.prefabPath; } public ItemParamData getCost(int numRolls) { return switch (numRolls) { case 10 -> new ItemParamData((costItemId10 > 0) ? costItemId10 : getCostItem(), costItemAmount10); default -> new ItemParamData(getCostItem(), costItemAmount * numRolls); }; } public int getCostItem() { return (costItem > 0) ? costItem : costItemId; } public int[] getRateUpItems4() { return (rateUpItems2.length > 0) ? rateUpItems2 : rateUpItems4; } public int[] getRateUpItems5() { return (rateUpItems1.length > 0) ? rateUpItems1 : rateUpItems5; } public boolean hasEpitomized() { return bannerType.equals(BannerType.WEAPON); } public int getWeight(int rarity, int pity) { return switch (rarity) { case 4 -> Utils.lerp(pity, weights4); default -> Utils.lerp(pity, weights5); }; } public int getPoolBalanceWeight(int rarity, int pity) { return switch (rarity) { case 4 -> Utils.lerp(pity, poolBalanceWeights4); default -> Utils.lerp(pity, poolBalanceWeights5); }; } public int getEventChance(int rarity) { return switch (rarity) { case 4 -> eventChance4; default -> (eventChance > -1) ? eventChance : eventChance5; }; } public GachaInfo toProto(Player player) { // TODO: use other Nonce/key insteadof session key to ensure the overall security for the player String sessionKey = player.getAccount().getSessionKey(); String record = "http" + (HTTP_ENCRYPTION.useInRouting ? "s" : "") + "://" + lr(HTTP_INFO.accessAddress, HTTP_INFO.bindAddress) + ":" + lr(HTTP_INFO.accessPort, HTTP_INFO.bindPort) + "/gacha?s=" + sessionKey + "&gachaType=" + gachaType; String details = "http" + (HTTP_ENCRYPTION.useInRouting ? "s" : "") + "://" + lr(HTTP_INFO.accessAddress, HTTP_INFO.bindAddress) + ":" + lr(HTTP_INFO.accessPort, HTTP_INFO.bindPort) + "/gacha/details?s=" + sessionKey + "&scheduleId=" + scheduleId; // Grasscutter.getLogger().info("record = " + record); ItemParamData costItem1 = this.getCost(1); ItemParamData costItem10 = this.getCost(10); PlayerGachaBannerInfo gachaInfo = player.getGachaInfo().getBannerInfo(this); int leftGachaTimes = switch (gachaTimesLimit) { case Integer.MAX_VALUE -> Integer.MAX_VALUE; default -> Math.max(gachaTimesLimit - gachaInfo.getTotalPulls(), 0); }; GachaInfo.Builder info = GachaInfo.newBuilder() .setGachaType(this.getGachaType()) .setScheduleId(this.getScheduleId()) .setBeginTime(this.getBeginTime()) .setEndTime(this.getEndTime()) .setCostItemId(costItem1.getId()) .setCostItemNum(costItem1.getCount()) .setTenCostItemId(costItem10.getId()) .setTenCostItemNum(costItem10.getCount()) .setGachaPrefabPath(this.getPrefabPath()) .setGachaPreviewPrefabPath(this.getPreviewPrefabPath()) .setGachaProbUrl(details) .setGachaProbUrlOversea(details) .setGachaRecordUrl(record) .setGachaRecordUrlOversea(record) .setLeftGachaTimes(leftGachaTimes) .setGachaTimesLimit(gachaTimesLimit) .setGachaSortId(this.getSortId()); if (hasEpitomized()) { info.setWishItemId(gachaInfo.getWishItemId()) .setWishProgress(gachaInfo.getFailedChosenItemPulls()) .setWishMaxProgress(this.getWishMaxProgress()); } if (this.getTitlePath() != null) { info.setTitleTextmap(this.getTitlePath()); } if (this.getRateUpItems5().length > 0) { GachaUpInfo.Builder upInfo = GachaUpInfo.newBuilder().setItemParentType(1); for (int id : getRateUpItems5()) { upInfo.addItemIdList(id); info.addDisplayUp5ItemList(id); } info.addGachaUpInfoList(upInfo); } if (this.getRateUpItems4().length > 0) { GachaUpInfo.Builder upInfo = GachaUpInfo.newBuilder().setItemParentType(2); for (int id : getRateUpItems4()) { upInfo.addItemIdList(id); if (info.getDisplayUp4ItemListCount() == 0) { info.addDisplayUp4ItemList(id); } } info.addGachaUpInfoList(upInfo); } return info.build(); } public enum BannerType { STANDARD, EVENT, WEAPON; } }