All Downloads are FREE. Search and download functionalities are using the official Maven repository.

personthecat.catlib.command.DefaultLibCommands Maven / Gradle / Ivy

package personthecat.catlib.command;

import com.google.common.collect.Lists;
import com.mojang.blaze3d.platform.Window;
import com.mojang.brigadier.arguments.StringArgumentType;
import lombok.experimental.UtilityClass;
import net.fabricmc.api.EnvType;
import net.fabricmc.api.Environment;
import net.minecraft.ChatFormatting;
import net.minecraft.Util;
import net.minecraft.client.Minecraft;
import net.minecraft.client.Options;
import net.minecraft.commands.Commands;
import net.minecraft.network.chat.*;
import net.minecraft.server.MinecraftServer;
import net.minecraft.world.effect.MobEffect;
import net.minecraft.world.effect.MobEffectInstance;
import net.minecraft.world.effect.MobEffects;
import net.minecraft.world.entity.player.Player;
import net.minecraft.world.level.GameType;
import personthecat.catlib.client.gui.SimpleTextPage;
import personthecat.catlib.command.arguments.JsonArgument;
import personthecat.catlib.config.LibConfig;
import personthecat.catlib.serialization.json.JsonPath;
import personthecat.catlib.data.ModDescriptor;
import personthecat.catlib.io.FileIO;
import personthecat.catlib.serialization.json.XjsUtils;
import personthecat.catlib.serialization.json.JsonCombiner;
import personthecat.catlib.util.McUtils;
import personthecat.catlib.util.PathUtils;
import xjs.core.Json;
import xjs.core.JsonObject;
import xjs.core.JsonValue;

import java.io.File;
import java.util.List;
import java.util.Objects;
import java.util.Optional;

import static net.minecraft.commands.Commands.m_82127_;
import static personthecat.catlib.command.CommandSuggestions.CURRENT_JSON;
import static personthecat.catlib.command.CommandUtils.arg;
import static personthecat.catlib.command.CommandUtils.fileArg;
import static personthecat.catlib.command.CommandUtils.greedyArg;
import static personthecat.catlib.command.CommandUtils.jsonFileArg;
import static personthecat.catlib.command.CommandUtils.jsonPathArg;
import static personthecat.catlib.util.PathUtils.extension;
import static personthecat.catlib.util.PathUtils.noExtension;

@UtilityClass
public class DefaultLibCommands {

    public static final String FILE_ARGUMENT = "file";
    public static final String TO_ARGUMENT = "to";
    public static final String PATH_ARGUMENT = "path";
    public static final String VALUE_ARGUMENT = "value";
    public static final String DIRECTORY_ARGUMENT = "dir";
    public static final String NAME_ARGUMENT = "name";
    public static final String MAX_ARGUMENT = "max";
    public static final String SCALE_ARGUMENT = "scale";

    /** The header displayed whenever the /display command runs. */
    private static final String DISPLAY_HEADER = "--- {} ---";

    /** The text formatting to be used for the display message header. */
    private static final Style DISPLAY_HEADER_STYLE = Style.f_131099_
        .m_131140_(ChatFormatting.GREEN)
        .m_131136_(true);

    /** The text formatting used to indicate values being deleted. */
    private static final Style DELETED_VALUE_STYLE = Style.f_131099_
        .m_131140_(ChatFormatting.RED);

    /** The text formatting used to indicate values being replaced. */
    private static final Style REPLACED_VALUE_STYLE = Style.f_131099_
        .m_131140_(ChatFormatting.GREEN);

    /** The text formatting used for the undo button. */
    private static final Style UNDO_STYLE = Style.f_131099_
        .m_131140_(ChatFormatting.GRAY)
        .m_131144_(new HoverEvent(HoverEvent.Action.f_130831_, new TextComponent("Click to undo.")))
        .m_131157_(ChatFormatting.UNDERLINE)
        .m_131136_(true);

    /** The number of backups before a warning is displayed. */
    private static final int BACKUP_COUNT_WARNING = 10;

    public static List createAll(final ModDescriptor mod, final boolean global) {
        return Lists.newArrayList(
            createDisplay(mod, global),
            createUpdate(mod, global),
            createBackup(mod, global),
            createCopy(mod, global),
            createMove(mod, global),
            createDelete(mod, global),
            createClean(mod, global),
            createRename(mod, global),
            createOpen(mod, global),
            createCombine(mod, global),
            createTest(mod, global),
            createUnTest(mod, global),
            createCh(mod, global),
            createCw(mod, global),
            createToJson(mod, global),
            createToXjs(mod, global)
        );
    }

    public static LibCommandBuilder createDisplay(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("display")
            .arguments(" []")
            .append("Outputs the contents of any JSON file to the chat.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(jsonFileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::display))
                .then(jsonPathArg(PATH_ARGUMENT)
                    .executes(utl.wrap(DefaultLibCommands::display))))
            );
    }

    public static LibCommandBuilder createUpdate(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("update")
            .arguments(" [] []")
            .append("Manually update a JSON value. Omit the value or")
            .append("path to display the current contents.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(jsonFileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::display))
                .then(jsonPathArg(PATH_ARGUMENT)
                    .executes(utl.wrap(DefaultLibCommands::display))
                .then(greedyArg(VALUE_ARGUMENT, CURRENT_JSON)
                    .executes(utl.wrap(DefaultLibCommands::update)))))
            );
    }

    public static LibCommandBuilder createBackup(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("backup")
            .arguments("")
            .append("Copies a file to the current mod's backup folder.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::backup)))
            );
    }

    public static LibCommandBuilder createCopy(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("copy")
            .arguments(" ")
            .append("Copies a file from one location to another.")
            .append("Accepts either a directory or a file as .")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                .then(fileArg(DIRECTORY_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::copy))))
            );
    }

    public static LibCommandBuilder createMove(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("move")
            .arguments(" ")
            .append("Moves a file from one location to another.")
            .append("Accepts either a directory or a file as .")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                .then(fileArg(DIRECTORY_ARGUMENT, mod)
                    .executes(utl.wrap( DefaultLibCommands::move))))
            );
    }

    public static LibCommandBuilder createDelete(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("delete")
            .arguments("")
            .append("Moves a file to the current mod's backup folder.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::delete)))
            );
    }

    public static LibCommandBuilder createClean(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("clean")
            .arguments("")
            .append("Moves all files in the given directory to the mod's")
            .append("backup folder. Else, deletes the contents of the")
            .append("mod's backup folder.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder.executes(utl.wrap(DefaultLibCommands::clean))
                .then(fileArg(DIRECTORY_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::clean)))
            );
    }

    public static LibCommandBuilder createRename(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("rename")
            .arguments(" ")
            .append("Renames a file without needing a path or extension.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                .then(Commands.m_82129_(NAME_ARGUMENT, StringArgumentType.word())
                    .executes(utl.wrap(DefaultLibCommands::rename))))
            );
    }

    public static LibCommandBuilder createOpen(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("open")
            .arguments("[]")
            .append("Opens a file or directory in your default editor.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::open)))
            );
    }

    public static LibCommandBuilder createCombine(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("combine")
            .arguments("  ")
            .append("Copies the given path from the first preset into the second preset.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(jsonFileArg(FILE_ARGUMENT, mod)
                .then(jsonPathArg(PATH_ARGUMENT)
                .then(jsonFileArg(TO_ARGUMENT, mod)
                    .executes(utl.wrap(DefaultLibCommands::combine)))))
            );
    }

    public static LibCommandBuilder createTest(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("test")
            .append("Applies night vision and spectator mode for easy cave viewing.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
                .generate((builder, utl) ->
                    builder.executes(utl.wrap(DefaultLibCommands::test)));
    }

    public static LibCommandBuilder createUnTest(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("untest")
            .append("Removes night vision and puts you in the default game mode.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) ->
                builder.executes(utl.wrap(DefaultLibCommands::unTest)));
    }

    public static LibCommandBuilder createCh(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("ch")
            .arguments("[]")
            .append("Allows you to expand your chat height beyond the")
            .append("default limit of 1.0.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .side(CommandSide.CLIENT)
            .generate((builder, utl) -> builder.executes(utl.wrap(ctx -> ch(ctx, false)))
                .then(m_82127_(MAX_ARGUMENT).executes(utl.wrap(ctx -> ch(ctx, true))))
                .then(arg(SCALE_ARGUMENT, 0.25, 5.0).executes(utl.wrap(ctx -> ch(ctx, false))))
            );
    }

    public static LibCommandBuilder createCw(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("cw")
            .arguments("[]")
            .append("Allows you to expand your chat width beyond the")
            .append("default limit of 1.0.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .side(CommandSide.CLIENT)
            .generate((builder, utl) -> builder.executes(utl.wrap(ctx -> cw(ctx, false)))
                .then(m_82127_(MAX_ARGUMENT).executes(utl.wrap(ctx -> cw(ctx, true))))
                .then(arg(SCALE_ARGUMENT, 0.25, 3.0).executes(utl.wrap(ctx -> cw(ctx, false))))
            );
    }

    public static LibCommandBuilder createToJson(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("tojson")
            .arguments("")
            .append("Converts an XJS file to a regular JSON file.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(ctx -> convert(ctx, true)))));
    }

    public static LibCommandBuilder createToXjs(final ModDescriptor mod, final boolean global) {
        return LibCommandBuilder.named("toxjs")
            .arguments("")
            .append("Converts a regular JSON into an XJS file.")
            .type(global ? CommandType.GLOBAL : CommandType.MOD)
            .generate((builder, utl) -> builder
                .then(fileArg(FILE_ARGUMENT, mod)
                    .executes(utl.wrap(ctx -> convert(ctx, false)))));
    }

    private static void display(final CommandContextWrapper wrapper) {
        final JsonArgument.Result file = wrapper.getJsonFile(FILE_ARGUMENT);
        final JsonValue json = wrapper.getOptional(PATH_ARGUMENT, JsonPath.class)
            .flatMap(result -> XjsUtils.getValueFromPath(file.json.get(), result))
            .orElseGet(file.json);

        final String header = file.file.getParentFile().getName() + "/" + file.file.getName();
        final Component headerComponent =
            wrapper.createText(DISPLAY_HEADER, header).m_6270_(DISPLAY_HEADER_STYLE);

        final String details = json.trim().toString(XjsUtils.noCr());
        final Component detailsComponent = wrapper.lintMessage(details);

        final long numLines = details.chars().filter(c -> c == '\n').count();
        if (McUtils.isClientSide() && numLines >= LibConfig.displayLength()) {
            loadTextPage(wrapper, headerComponent, detailsComponent);
            return;
        }
        wrapper.generateMessage("")
            .append(headerComponent)
            .append("\n")
            .append(detailsComponent)
            .sendMessage();
    }

    @Environment(EnvType.CLIENT)
    private static void loadTextPage(
            final CommandContextWrapper wrapper, final Component header, final Component details) {
        wrapper.setScreen(new SimpleTextPage(null, header, details));
    }

    private static void update(final CommandContextWrapper wrapper) {
        final JsonArgument.Result file = wrapper.getJsonFile(FILE_ARGUMENT);
        final JsonPath path = wrapper.getJsonPath(PATH_ARGUMENT);

        // Read the old and new values.
        final String toEscaped = wrapper.getString(VALUE_ARGUMENT);
        final String toLiteral = unEscape(toEscaped);
        final JsonValue toValue = Json.parse(toLiteral).trim();
        final JsonValue fromValue = XjsUtils.getValueFromPath(file.json.get(), path)
            .orElseGet(() -> Json.value(null));
        final String fromLiteral = fromValue.unformatted().toString(XjsUtils.noCr());
        final String fromEscaped = escape(fromLiteral);

        // Write the new value.
        XjsUtils.setValueFromPath(file.json.get(), path, toValue.isNull() ? null : toValue);
        XjsUtils.writeJson(file.json.get(), file.file)
            .expect("Error writing to file: {}", file.file.getName());

        // Send feedback.
        wrapper.generateMessage("Successfully updated {}.\n", file.file.getName())
            .append(fromLiteral.replace("\r", ""), DELETED_VALUE_STYLE)
            .append(" -> ")
            .append(toLiteral, REPLACED_VALUE_STYLE)
            .append(" ")
            .append(generateUndo(wrapper.getInput(), fromEscaped, toEscaped))
            .sendMessage();
    }

    private static TextComponent generateUndo(final String input, final String from, final String to) {
        final int index = input.lastIndexOf(to);
        final String command = (input.substring(0, index) + from).replace("\"\"", "\"");
        final ClickEvent undo = new ClickEvent(ClickEvent.Action.RUN_COMMAND, command);
        return (TextComponent) new TextComponent("[UNDO]").m_6270_(UNDO_STYLE.m_131142_(undo));
    }

    private static String escape(final String literal) {
        return literal.replaceAll("\r?\n", "\\\\n") // newline -> \n
            .replace("\"", "\\\""); // " -> \"
    }

    private static String unEscape(final String escaped) {
        return escaped.replace("\\n", "\n")
            .replace("\\\"", "\"");
    }

    private static void backup(final CommandContextWrapper wrapper) {
        if (BACKUP_COUNT_WARNING < FileIO.backup(wrapper.getBackupsFolder(), wrapper.getFile(FILE_ARGUMENT), true)) {
            wrapper.sendError("{} backups detected. Consider cleaning these out.", BACKUP_COUNT_WARNING);
        }
        wrapper.sendMessage("Backup created successfully.");
    }

    private static void copy(final CommandContextWrapper wrapper) {
        FileIO.copy(wrapper.getFile(FILE_ARGUMENT), wrapper.getFile(DIRECTORY_ARGUMENT))
            .expect("The file could not be copied.");
        wrapper.sendMessage("File copied successfully.");
    }

    private static void move(final CommandContextWrapper wrapper) {
        FileIO.move(wrapper.getFile(FILE_ARGUMENT), wrapper.getFile(DIRECTORY_ARGUMENT))
            .expect("The file could not be moved.");
        wrapper.sendMessage("File moved successfully.");
    }

    private static void delete(final CommandContextWrapper wrapper) {
        final File file = wrapper.getFile(FILE_ARGUMENT);
        if (PathUtils.isIn(wrapper.getMod().getBackupFolder(), file)) {
            FileIO.delete(file);
            wrapper.sendMessage("File deleted successfully.");
        } else if (BACKUP_COUNT_WARNING < FileIO.backup(wrapper.getBackupsFolder(), wrapper.getFile(FILE_ARGUMENT), false)) {
            wrapper.sendError("{} backups have been created. Consider cleaning these out.", BACKUP_COUNT_WARNING);
        }
        wrapper.sendMessage("File moved to backups.");
    }

    private static void clean(final CommandContextWrapper wrapper) {
        final Optional directory = wrapper.getOptional(DIRECTORY_ARGUMENT, File.class)
            .filter(dir -> !dir.equals(wrapper.getBackupsFolder()));
        if (directory.isPresent()) {
            final File dir = directory.get();
            if (dir.isFile()) {
                if (!dir.delete()) {
                    wrapper.sendError("Error deleting {}", dir.getName());
                } else {
                    wrapper.sendMessage("Deleted {}", dir.getName());
                }
            } else {
                cleanFiles(wrapper, dir);
            }
        } else {
            cleanFiles(wrapper, wrapper.getBackupsFolder());
        }
    }

    private static void cleanFiles(final CommandContextWrapper wrapper, final File dir) {
        final int deleted = deleteNotRecursive(dir);
        if (deleted < 0) {
            wrapper.sendError("Some files could not be deleted.");
        } else {
            wrapper.sendMessage("Deleted {} files.", deleted);
        }
    }

    private static int deleteNotRecursive(final File dir) {
        int deleted = 0;
        for (final File f : FileIO.listFiles(dir)) {
            if (f.isFile() && !f.delete()) {
                return -1;
            }
            deleted++;
        }
        return deleted;
    }

    private static void rename(final CommandContextWrapper wrapper) {
        FileIO.rename(wrapper.getFile(FILE_ARGUMENT), wrapper.getString(NAME_ARGUMENT))
            .expect("Error renaming file.");
        wrapper.sendMessage("File renamed successfully.");
    }

    private static void open(final CommandContextWrapper wrapper) {
        Util.m_137581_().m_137644_(wrapper.getFile(FILE_ARGUMENT));
    }

    private static void combine(final CommandContextWrapper wrapper) {
        final JsonArgument.Result from = wrapper.getJsonFile(FILE_ARGUMENT);
        final JsonPath path = wrapper.getJsonPath(PATH_ARGUMENT);
        final JsonArgument.Result to = wrapper.getJsonFile(TO_ARGUMENT);

        if (BACKUP_COUNT_WARNING < FileIO.backup(wrapper.getBackupsFolder(), to.file, true)) {
            wrapper.sendError("Created > {} backups. Consider cleaning these out.", BACKUP_COUNT_WARNING);
        }
        JsonCombiner.combine(from, to, path);
        wrapper.sendMessage("Finished combining file. The original was moved to the backups directory.");
    }

    private static void test(final CommandContextWrapper wrapper) {
        final Player player = wrapper.getPlayer();
        if (player != null) {
            wrapper.setGameMode(GameType.SPECTATOR);
            final MobEffect nightVision = MobEffects.f_19611_;
            player.m_7292_(new MobEffectInstance(nightVision, 999999999, 1, true, false));
        }
    }

    private static void unTest(final CommandContextWrapper wrapper) {
        final Player player = wrapper.getPlayer();
        if (player != null) {
            final GameType mode = Optional.ofNullable(wrapper.getServer())
                .map(MinecraftServer::m_130008_)
                .orElse(GameType.CREATIVE);

            wrapper.setGameMode(mode);
            player.m_21195_(MobEffects.f_19611_);
        }
    }

    private static void ch(final CommandContextWrapper wrapper, final boolean max) {
        final Minecraft mc = Minecraft.m_91087_();
        final Options cfg = mc.f_91066_;
        final Window window = Objects.requireNonNull(mc.m_91268_(), "Not running client side");
        final Optional scaleArgument = wrapper.getOptional(SCALE_ARGUMENT, Double.class);

        if (!max && scaleArgument.isEmpty()) {
            wrapper.sendMessage("Current chat height: {}", cfg.f_92134_);
            return;
        }

        final double possible = (window.m_85446_() - 20.0) / 180.0;
        if (max) {
            cfg.f_92134_ = possible;
        } else {
            final double height = (scaleArgument.get() * 180.0 + 20.0) * window.m_85449_();
            if (height > window.m_85442_()) {
                wrapper.sendError("Max size is {} with your current window and scale.", possible);
                return;
            }
            cfg.f_92134_ = scaleArgument.get();
        }
        cfg.m_92169_();

        wrapper.sendMessage("Updated chat height: {}", cfg.f_92134_);
    }

    private static void cw(final CommandContextWrapper wrapper, final boolean max) {
        final Minecraft mc = Minecraft.m_91087_();
        final Options cfg = mc.f_91066_;
        final Window window = Objects.requireNonNull(mc.m_91268_(), "Not running client side");
        final Optional scaleArgument = wrapper.getOptional(SCALE_ARGUMENT, Double.class);

        if (!max && scaleArgument.isEmpty()) {
            wrapper.sendMessage("Current chat width: {}", cfg.f_92132_);
            return;
        }

        final double possible = (double) window.m_85445_() / 320.0;
        if (max) {
            cfg.f_92132_ = possible;
        } else {
            final double width = scaleArgument.get() * 320.0 * window.m_85449_();
            if (width > window.m_85441_()) {
                wrapper.sendError("Max size is {} with your current window and scale.", possible);
                return;
            }
            cfg.f_92132_ = scaleArgument.get();
        }
        cfg.m_92169_();

        wrapper.sendMessage("Updated chat width: {}", cfg.f_92132_);
    }

    private static void convert(final CommandContextWrapper wrapper, final boolean toJson) {
        final File source = wrapper.getFile(FILE_ARGUMENT);
        if (!source.exists()) {
            wrapper.sendError("Found not found.");
            return;
        }
        if (toJson == "json".equals(extension(source))) {
            wrapper.sendError("File is already in the desired format.");
            return;
        }
        final Optional json = XjsUtils.readJson(source);
        if (json.isEmpty()) {
            wrapper.sendError("The file could not be read.");
            return;
        }
        final String extension = toJson ? "json" : "xjs";
        final File converted = new File(source.getParentFile(), noExtension(source) + extension);
        XjsUtils.writeJson(json.get(), converted).expect("Error writing file.");

        if (!PathUtils.isIn(wrapper.getBackupsFolder(), source)) {
            FileIO.backup(wrapper.getBackupsFolder(), source, false);
            wrapper.sendMessage("File converted successfully. The original was moved to the backups directory.");
        } else {
            wrapper.sendMessage("File converted successfully. The original could not be backed up.");
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy