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

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

import com.google.gson.reflect.TypeToken;

17
import emu.grasscutter.GameConstants;
Melledy's avatar
Melledy committed
18
import emu.grasscutter.Grasscutter;
19
import emu.grasscutter.command.CommandHandler;
20
import emu.grasscutter.command.CommandMap;
21
import emu.grasscutter.data.GameData;
Melledy's avatar
Melledy committed
22
import emu.grasscutter.data.ResourceLoader;
Melledy's avatar
Melledy committed
23
24
25
import emu.grasscutter.data.excels.AvatarData;
import emu.grasscutter.data.excels.ItemData;
import emu.grasscutter.data.excels.QuestData;
AnimeGitB's avatar
AnimeGitB committed
26
import emu.grasscutter.utils.Language;
27
import emu.grasscutter.utils.Utils;
AnimeGitB's avatar
AnimeGitB committed
28
import emu.grasscutter.utils.Language.TextStrings;
AnimeGitB's avatar
AnimeGitB committed
29
30
31
import it.unimi.dsi.fastutil.ints.Int2IntMap;
import it.unimi.dsi.fastutil.ints.Int2IntOpenHashMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
Melledy's avatar
Melledy committed
32

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

KingRainbow44's avatar
KingRainbow44 committed
35
public final class Tools {
AnimeGitB's avatar
AnimeGitB committed
36
    public static void createGmHandbooks() throws Exception {
37
38
39
        Language savedLanguage = Grasscutter.getLanguage();
        Int2ObjectMap<TextStrings> textMaps = Language.getTextMapStrings();

AnimeGitB's avatar
AnimeGitB committed
40
41
42
43
44
        ResourceLoader.loadAll();
        Int2IntMap avatarNames = new Int2IntOpenHashMap(GameData.getAvatarDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        Int2IntMap itemNames = new Int2IntOpenHashMap(GameData.getItemDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        Int2IntMap monsterNames = new Int2IntOpenHashMap(GameData.getMonsterDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getNameTextMapHash())));
        Int2IntMap mainQuestTitles = new Int2IntOpenHashMap(GameData.getMainQuestDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getTitleTextMapHash())));
AnimeGitB's avatar
AnimeGitB committed
45
46
        // Int2IntMap questDescs = new Int2IntOpenHashMap(GameData.getQuestDataMap().int2ObjectEntrySet().stream().collect(Collectors.toMap(e -> (int) e.getIntKey(), e -> (int) e.getValue().getDescTextMapHash())));

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

        // Write txt files
        for (int i = 0; i < TextStrings.NUM_LANGUAGES; i++) {
            String fileName = "./GM Handbook - %s.txt".formatted(TextStrings.ARR_LANGUAGES[i]);
            try (PrintWriter writer = new PrintWriter(new OutputStreamWriter(new FileOutputStream(fileName), StandardCharsets.UTF_8), false)) {
108
                writer.write(handbookBuilders.get(i).toString());
AnimeGitB's avatar
AnimeGitB committed
109
110
111
112
113
            }
        }
        Grasscutter.getLogger().info("GM Handbooks generated!");
    }

github-actions's avatar
github-actions committed
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
    public static void createGachaMapping(String location) throws Exception {
        ToolsWithLanguageOption.createGachaMapping(location, getLanguageOption());
    }

    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;
    }

    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");
        }

        stagedMessage.append("\nYour choice:[EN] ");

        input = Grasscutter.getConsole().readLine(stagedMessage.toString());
        if (availableLangList.contains(input.toLowerCase())) {
            return input.toUpperCase();
        }
        Grasscutter.getLogger().info("Invalid option. Will use EN(English) as fallback");

        return "EN";
    }
164
165
166
}

final class ToolsWithLanguageOption {
github-actions's avatar
github-actions committed
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
    @SuppressWarnings("deprecation")
    public static void createGachaMapping(String location, String language) throws Exception {
        ResourceLoader.loadResources();

        Map<Long, String> map;
        try (InputStreamReader fileReader = new InputStreamReader(new FileInputStream(Utils.toFilePath(RESOURCE("TextMap/TextMap" + language + ".json"))), StandardCharsets.UTF_8)) {
            map = Grasscutter.getGsonFactory().fromJson(fileReader, new TypeToken<Map<Long, String>>() {}.getType());
        }

        List<Integer> list;

        try (PrintWriter writer = new PrintWriter(new OutputStreamWriter(new FileOutputStream(location), StandardCharsets.UTF_8), false)) {
            list = new ArrayList<>(GameData.getAvatarDataMap().keySet());
            Collections.sort(list);

            // 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.
            // The enduser can still modify the `gacha_mappings.js` directly to enable multilingual for the gacha record system.
            writer.println("mappings = {\"en-us\": {");

            // Avatars
            boolean first = true;
            for (Integer id : list) {
                AvatarData data = GameData.getAvatarDataMap().get(id);
                int avatarID = data.getId();
                if (avatarID >= 11000000) { // skip test avatar
                    continue;
                }
                if (first) { // skip adding comma for the first element
                    first = false;
                } else {
                    writer.print(",");
                }
                String color = switch (data.getQualityType()) {
                    case "QUALITY_PURPLE" -> "purple";
                    case "QUALITY_ORANGE" -> "yellow";
                    default -> "blue";
                };
                // Got the magic number 4233146695 from manually search in the json file
                writer.println(
                    "\"" + (avatarID % 1000 + 1000) + "\" : [\""
                    + map.get(data.getNameTextMapHash()) + "(" +  map.get(4233146695L)+ ")\", \""
                    + color + "\"]");
            }

            writer.println();

            list = new ArrayList<>(GameData.getItemDataMap().keySet());
            Collections.sort(list);

            // Weapons
            for (Integer id : list) {
                ItemData data = GameData.getItemDataMap().get(id);
                if (data.getId() <= 11101 || data.getId() >= 20000) {
                    continue; //skip non weapon items
                }
                String color;

                switch (data.getRankLevel()) {
                    case 3:
                        color = "blue";
                        break;
                    case 4:
                        color = "purple";
                        break;
                    case 5:
                        color = "yellow";
                        break;
                    default:
                        continue; // skip unnecessary entries
                }

                // Got the magic number 4231343903 from manually search in the json file

                writer.println(",\"" + data.getId() +
                         "\" : [\"" + map.get(data.getNameTextMapHash()).replaceAll("\"", "")
                         + "("+ map.get(4231343903L)+")\",\""+ color + "\"]");
            }
            writer.println(",\"200\": \""+map.get(332935371L)+"\", \"301\": \""+ map.get(2272170627L) + "\", \"302\": \""+map.get(2864268523L)+"\"");
            writer.println("}\n}");
        }

        Grasscutter.getLogger().info("Mappings generated to " + location + " !");
    }
Melledy's avatar
Melledy committed
251
}