Tools.java 13.5 KB
Newer Older
Melledy's avatar
Melledy committed
1
2
package emu.grasscutter.tools;

3
import java.io.File;
4
5
import java.io.FileOutputStream;
import java.io.OutputStreamWriter;
Melledy's avatar
Melledy committed
6
import java.io.PrintWriter;
7
import java.nio.charset.StandardCharsets;
Melledy's avatar
Melledy committed
8
9
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
10
import java.util.*;
AnimeGitB's avatar
AnimeGitB committed
11
import java.util.stream.Collectors;
Melledy's avatar
Melledy committed
12

13
import emu.grasscutter.GameConstants;
Melledy's avatar
Melledy committed
14
import emu.grasscutter.Grasscutter;
15
import emu.grasscutter.command.CommandHandler;
16
import emu.grasscutter.command.CommandMap;
17
import emu.grasscutter.data.GameData;
Melledy's avatar
Melledy committed
18
import emu.grasscutter.data.ResourceLoader;
Melledy's avatar
Melledy committed
19
20
21
import emu.grasscutter.data.excels.AvatarData;
import emu.grasscutter.data.excels.ItemData;
import emu.grasscutter.data.excels.QuestData;
AnimeGitB's avatar
AnimeGitB committed
22
import emu.grasscutter.utils.Language;
AnimeGitB's avatar
AnimeGitB committed
23
import emu.grasscutter.utils.Language.TextStrings;
AnimeGitB's avatar
AnimeGitB committed
24
25
import it.unimi.dsi.fastutil.ints.Int2IntSortedMap;
import it.unimi.dsi.fastutil.ints.Int2IntRBTreeMap;
AnimeGitB's avatar
AnimeGitB committed
26
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
Melledy's avatar
Melledy committed
27

28
import static emu.grasscutter.config.Configuration.*;
方块君's avatar
方块君 committed
29

KingRainbow44's avatar
KingRainbow44 committed
30
public final class Tools {
AnimeGitB's avatar
AnimeGitB committed
31
    public static void createGmHandbooks() throws Exception {
32
33
        final List<Language> languages = Language.TextStrings.getLanguages();
        final Int2ObjectMap<TextStrings> textMaps = Language.getTextMapStrings();
34

AnimeGitB's avatar
AnimeGitB committed
35
        ResourceLoader.loadAll();
AnimeGitB's avatar
AnimeGitB committed
36
37
38
39
40
        final Int2IntSortedMap avatarNames = new Int2IntRBTreeMap(GameData.getAvatarDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        final Int2IntSortedMap itemNames = new Int2IntRBTreeMap(GameData.getItemDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        final Int2IntSortedMap monsterNames = new Int2IntRBTreeMap(GameData.getMonsterDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        final Int2IntSortedMap mainQuestTitles = new Int2IntRBTreeMap(GameData.getMainQuestDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getTitleTextMapHash())));
        // Int2IntSortedMap questDescs = new Int2IntRBTreeMap(GameData.getQuestDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getDescTextMapHash())));
AnimeGitB's avatar
AnimeGitB committed
41

AnimeGitB's avatar
AnimeGitB committed
42
        // Preamble
43
44
        final List<StringBuilder> handbookBuilders = new ArrayList<>(TextStrings.NUM_LANGUAGES);
        final String now = DateTimeFormatter.ofPattern("yyyy/MM/dd HH:mm:ss").format(LocalDateTime.now());
45
46
        for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++)
            handbookBuilders.add(new StringBuilder()
AnimeGitB's avatar
AnimeGitB committed
47
48
                .append("// Grasscutter " + GameConstants.VERSION + " GM Handbook\n")
                .append("// Created " + now + "\n\n")
49
                .append("// Commands\n"));
AnimeGitB's avatar
AnimeGitB committed
50
        // Commands
51
        final List<CommandHandler> cmdList = CommandMap.getInstance().getHandlersAsList();
52
        final String padCmdLabel = "%" + cmdList.stream().map(CommandHandler::getLabel).map(String::length).max(Integer::compare).get().toString() + "s : ";
AnimeGitB's avatar
AnimeGitB committed
53
        for (CommandHandler cmd : cmdList) {
54
55
            final String label = padCmdLabel.formatted(cmd.getLabel());
            final String descKey = cmd.getDescriptionKey();
AnimeGitB's avatar
AnimeGitB committed
56
            for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++) {
57
                String desc = languages.get(i).get(descKey).replace("\n", "\n\t\t\t\t").replace("\t", "    ");
58
                handbookBuilders.get(i).append(label + desc + "\n");
AnimeGitB's avatar
AnimeGitB committed
59
60
            }
        }
61
        // Avatars, Items, Monsters
62
        final String[] handbookSections = {"Avatars", "Items", "Monsters"};
AnimeGitB's avatar
AnimeGitB committed
63
        final Int2IntSortedMap[] handbookNames = {avatarNames, itemNames, monsterNames};
64
        for (int section = 0; section < handbookSections.length; section++) {
AnimeGitB's avatar
AnimeGitB committed
65
            final var h = handbookNames[section];
66
67
            final String s = "\n\n// " + handbookSections[section] + "\n";
            handbookBuilders.forEach(b -> b.append(s));
AnimeGitB's avatar
AnimeGitB committed
68
69
            final String padId = "%" + Integer.toString(h.keySet().lastInt()).length() + "s : ";
            h.forEach((id, hash) -> {
70
                final String sId = padId.formatted(id);
AnimeGitB's avatar
AnimeGitB committed
71
                final TextStrings t = textMaps.get((int) hash);
72
73
74
                for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++)
                    handbookBuilders.get(i).append(sId + t.strings[i] + "\n");
            });
AnimeGitB's avatar
AnimeGitB committed
75
76
        }
        // Scenes - no translations
77
        handbookBuilders.forEach(b -> b.append("\n\n// Scenes\n"));
78
        final var sceneDataMap = GameData.getSceneDataMap();
79
        final String padSceneId = "%" + Integer.toString(sceneDataMap.keySet().intStream().max().getAsInt()).length() + "d : ";
AnimeGitB's avatar
AnimeGitB committed
80
        sceneDataMap.keySet().intStream().sorted().forEach(id -> {
81
82
            final String sId = padSceneId.formatted(id);
            final String data = sceneDataMap.get(id).getScriptData();
83
            handbookBuilders.forEach(b -> b.append(sId + data + "\n"));
AnimeGitB's avatar
AnimeGitB committed
84
85
        });
        // Quests
86
        handbookBuilders.forEach(b -> b.append("\n\n// Quests\n"));
87
        final var questDataMap = GameData.getQuestDataMap();
88
        final String padQuestId = "%" + Integer.toString(questDataMap.keySet().intStream().max().getAsInt()).length() + "d : ";
AnimeGitB's avatar
AnimeGitB committed
89
        questDataMap.keySet().intStream().sorted().forEach(id -> {
90
91
92
93
            final String sId = padQuestId.formatted(id);
            final QuestData data = questDataMap.get(id);
            final TextStrings title = textMaps.get((int) mainQuestTitles.get(data.getMainId()));
            final TextStrings desc = textMaps.get((int) data.getDescTextMapHash());
94
95
            for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++)
                handbookBuilders.get(i).append(sId + title.strings[i] + " - " + desc.strings[i] + "\n");
AnimeGitB's avatar
AnimeGitB committed
96
97
98
99
        });

        // Write txt files
        for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++) {
100
            final String fileName = "./GM Handbook - %s.txt".formatted(TextStrings.ARR_LANGUAGES[i]);
AnimeGitB's avatar
AnimeGitB committed
101
            try (PrintWriter writer = new PrintWriter(new OutputStreamWriter(new FileOutputStream(fileName), StandardCharsets.UTF_8), false)) {
102
                writer.write(handbookBuilders.get(i).toString());
AnimeGitB's avatar
AnimeGitB committed
103
104
105
106
107
            }
        }
        Grasscutter.getLogger().info("GM Handbooks generated!");
    }

github-actions's avatar
github-actions committed
108
    public static void createGachaMapping(String location) throws Exception {
AnimeGitB's avatar
AnimeGitB committed
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
        createGachaMappings(location);
    }

    public static List<String> createGachaMappingJsons() {
        final int NUM_LANGUAGES = Language.TextStrings.NUM_LANGUAGES;
        final Language.TextStrings CHARACTER = Language.getTextMapKey(4233146695L);  // "Character" in EN
        final Language.TextStrings WEAPON = Language.getTextMapKey(4231343903L);  // "Weapon" in EN
        final Language.TextStrings STANDARD_WISH = Language.getTextMapKey(332935371L);  // "Standard Wish" in EN
        final Language.TextStrings CHARACTER_EVENT_WISH = Language.getTextMapKey(2272170627L);  // "Character Event Wish" in EN
        final Language.TextStrings CHARACTER_EVENT_WISH_2 = Language.getTextMapKey(3352513147L);  // "Character Event Wish-2" in EN
        final Language.TextStrings WEAPON_EVENT_WISH = Language.getTextMapKey(2864268523L);  // "Weapon Event Wish" in EN
        final List<StringBuilder> sbs = new ArrayList<>(NUM_LANGUAGES);
        for (int langIdx = 0; langIdx < NUM_LANGUAGES; langIdx++)
            sbs.add(new StringBuilder("{\n"));  // Web requests should never need Windows line endings

        // Avatars
        GameData.getAvatarDataMap().keySet().intStream().sorted().forEach(id -> {
            AvatarData data = GameData.getAvatarDataMap().get(id);
            int avatarID = data.getId();
            if (avatarID >= 11000000) { // skip test avatar
                return;
            }
            String color = switch (data.getQualityType()) {
                case "QUALITY_PURPLE" -> "purple";
                case "QUALITY_ORANGE" -> "yellow";
                case "QUALITY_BLUE" -> "blue";
                default -> "";
            };
            Language.TextStrings avatarName = Language.getTextMapKey(data.getNameTextMapHash());
            for (int langIdx = 0; langIdx < NUM_LANGUAGES; langIdx++) {
                sbs.get(langIdx)
                    .append("\t\"")
                    .append(avatarID % 1000 + 1000)
                    .append("\": [\"")
                    .append(avatarName.get(langIdx))
                    .append(" (")
                    .append(CHARACTER.get(langIdx))
                    .append(")\", \"")
                    .append(color)
                    .append("\"],\n");
            }
        });

        // Weapons
        GameData.getItemDataMap().keySet().intStream().sorted().forEach(id -> {
            ItemData data = GameData.getItemDataMap().get(id);
            if (data.getId() <= 11101 || data.getId() >= 20000) {
                return; //skip non weapon items
            }
            String color = switch (data.getRankLevel()) {
                case 3 -> "blue";
                case 4 -> "purple";
                case 5 -> "yellow";
                default -> null;
            };
            if (color == null) return;  // skip unnecessary entries
            Language.TextStrings weaponName = Language.getTextMapKey(data.getNameTextMapHash());
            for (int langIdx = 0; langIdx < NUM_LANGUAGES; langIdx++) {
                sbs.get(langIdx)
                    .append("\t\"")
                    .append(data.getId())
                    .append("\": [\"")
                    .append(weaponName.get(langIdx).replaceAll("\"", "\\\\\""))
                    .append(" (")
                    .append(WEAPON.get(langIdx))
                    .append(")\", \"")
                    .append(color)
                    .append("\"],\n");
            }
        });

        for (int langIdx = 0; langIdx < NUM_LANGUAGES; langIdx++) {
            sbs.get(langIdx)
                .append("\t\"200\": \"")
                .append(STANDARD_WISH.get(langIdx))
                .append("\",\n\t\"301\": \"")
                .append(CHARACTER_EVENT_WISH.get(langIdx))
                .append("\",\n\t\"400\": \"")
                .append(CHARACTER_EVENT_WISH_2.get(langIdx))
                .append("\",\n\t\"302\": \"")
                .append(WEAPON_EVENT_WISH.get(langIdx))
                .append("\"\n}");
        }
        return sbs.stream().map(StringBuilder::toString).toList();
    }

    public static void createGachaMappings(String location) throws Exception {
        ResourceLoader.loadResources();
        List<String> jsons = createGachaMappingJsons();
        StringBuilder sb = new StringBuilder("mappings = {\n");
        for (int i = 0; i < Language.TextStrings.NUM_LANGUAGES; i++) {
            sb.append("\t\"%s\": ".formatted(Language.TextStrings.ARR_GC_LANGUAGES[i].toLowerCase()));  // TODO: change the templates to not use lowercased locale codes
            sb.append(jsons.get(i).replace("\n", "\n\t") + ",\n");
        }
        sb.setLength(sb.length() - 2);  // Delete trailing ",\n"
        sb.append("\n}");

        try (PrintWriter writer = new PrintWriter(new OutputStreamWriter(new FileOutputStream(location), StandardCharsets.UTF_8), false)) {
            // if the user made choices for language, I assume it's okay to assign his/her selected language to "en-us"
            // since it's the fallback language and there will be no difference in the gacha record page.
209
            // The end-user can still modify the `gacha/mappings.js` directly to enable multilingual for the gacha record system.
AnimeGitB's avatar
AnimeGitB committed
210
211
212
            writer.println(sb);
            Grasscutter.getLogger().info("Mappings generated to " + location + " !");
        }
github-actions's avatar
github-actions committed
213
214
215
216
217
218
219
220
221
222
    }

    public static List<String> getAvailableLanguage() {
        File textMapFolder = new File(RESOURCE("TextMap"));
        List<String> availableLangList = new ArrayList<>();
        for (String textMapFileName : Objects.requireNonNull(textMapFolder.list((dir, name) -> name.startsWith("TextMap") && name.endsWith(".json")))) {
            availableLangList.add(textMapFileName.replace("TextMap", "").replace(".json", "").toLowerCase());
        } return availableLangList;
    }

223
    @Deprecated(forRemoval = true, since = "1.2.3")
github-actions's avatar
github-actions committed
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
    public static String getLanguageOption() {
        List<String> availableLangList = getAvailableLanguage();

        // Use system out for better format
        if (availableLangList.size() == 1) {
            return availableLangList.get(0).toUpperCase();
        }
        StringBuilder stagedMessage = new StringBuilder();
        stagedMessage.append("The following languages mappings are available, please select one: [default: EN] \n");

        StringBuilder groupedLangList = new StringBuilder(">\t"); String input;
        int groupedLangCount = 0;

        for (String availableLanguage: availableLangList) {
            groupedLangCount++;
            groupedLangList.append(availableLanguage).append("\t");

            if (groupedLangCount == 6) {
                stagedMessage.append(groupedLangList).append("\n");
                groupedLangCount = 0;
                groupedLangList = new StringBuilder(">\t");
            }
        }

        if (groupedLangCount > 0) {
            stagedMessage.append(groupedLangList).append("\n");
        }

252
        stagedMessage.append("\nYour choice: [EN] ");
github-actions's avatar
github-actions committed
253
254
255
256
257
258

        input = Grasscutter.getConsole().readLine(stagedMessage.toString());
        if (availableLangList.contains(input.toLowerCase())) {
            return input.toUpperCase();
        }

259
        Grasscutter.getLogger().info("Invalid option. Will use EN (English) as fallback."); return "EN";
github-actions's avatar
github-actions committed
260
    }
261
}