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

cn.nukkit.level.generator.Generator Maven / Gradle / Ivy

There is a newer version: 1.20.40-r1
Show newest version
package cn.nukkit.level.generator;

import cn.nukkit.Server;
import cn.nukkit.api.PowerNukkitXDifference;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import cn.nukkit.block.BlockID;
import cn.nukkit.level.ChunkManager;
import cn.nukkit.level.DimensionData;
import cn.nukkit.level.DimensionEnum;
import cn.nukkit.level.Level;
import cn.nukkit.level.generator.populator.type.PopulatorStructure;
import cn.nukkit.level.generator.task.ChunkPopulationTask;
import cn.nukkit.math.NukkitRandom;
import cn.nukkit.math.Vector3;
import cn.nukkit.scheduler.AsyncTask;

import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

/**
 * @author MagicDroidX (Nukkit Project)
 */
public abstract class Generator implements BlockID {

    public static final int TYPE_OLD = 0;
    public static final int TYPE_INFINITE = 1;
    public static final int TYPE_FLAT = 2;
    public static final int TYPE_NETHER = 3;
    public static final int TYPE_THE_END = 4;

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    protected Level level;

    @Since("1.19.21-r2")
    protected ChunkManager chunkManager;

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    protected NukkitRandom random;

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    protected List structurePopulators = new ArrayList<>();

    {
        if (shouldGenerateStructures()) {
            try {
                for (Class cz : PopulatorStructure.getPopulators()) {
                    structurePopulators.add(cz.getConstructor().newInstance());
                }
            } catch (InstantiationException | NoSuchMethodException | InvocationTargetException |
                     IllegalAccessException e) {
                throw new RuntimeException(e);
            }
        }
    }

    public abstract int getId();

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public List getStructurePopulators() {
        return structurePopulators;
    }

    public DimensionData getDimensionData() {
        DimensionData dimensionData = DimensionEnum.getDataFromId(this.getDimension());
        if (dimensionData == null) {
            dimensionData = DimensionEnum.OVERWORLD.getDimensionData();
        }
        return dimensionData;
    }

    /**
     * 返回此生成器实例绑定的世界
     * 你不应该将此方法的返回值用于{@link cn.nukkit.level.generator.populator.type.Populator}上,而是使用{@code getChunkManager()}方法
     * 以更好地利用多线程
     *
     * @return {@link cn.nukkit.level.generator.populator.type.Populator}
     */
    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public Level getLevel() {
        return level;
    }

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public NukkitRandom getRandom() {
        return random;
    }

    /**
     * 返回生成器的目标区块管理器
     * 实际为{@link PopChunkManager}
     *
     * @return {@link ChunkManager}
     */
    @PowerNukkitXDifference
    @Since("1.19.21-r2")
    public ChunkManager getChunkManager() {
        return chunkManager;
    }

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public void setLevel(Level level) {
        this.level = level;
    }

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public void setChunkManager(ChunkManager chunkManager) {
        this.chunkManager = chunkManager;
    }

    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public void setRandom(NukkitRandom random) {
        this.random = random;
    }

    @Deprecated
    public int getDimension() {
        return Level.DIMENSION_OVERWORLD;
    }

    private static final Map> nameList = new HashMap<>();

    private static final Map> typeList = new HashMap<>();

    public static boolean addGenerator(Class clazz, String name, int type) {
        name = name.toLowerCase();
        if (clazz != null && !Generator.nameList.containsKey(name)) {
            Generator.nameList.put(name, clazz);
            if (!Generator.typeList.containsKey(type)) {
                Generator.typeList.put(type, clazz);
            }
            return true;
        }
        return false;
    }

    /**
     * 注册未知类型的生成器(Terra)
     */
    @PowerNukkitXOnly
    @Since("1.19.50-r1")
    public static boolean addGenerator(Class clazz, String name) {
        name = name.toLowerCase();
        if (clazz != null && !Generator.nameList.containsKey(name)) {
            Generator.nameList.put(name, clazz);
            return true;
        }
        return false;
    }

    public static String[] getGeneratorList() {
        String[] keys = new String[Generator.nameList.size()];
        return Generator.nameList.keySet().toArray(keys);
    }

    public static Class getGenerator(String name) {
        name = name.toLowerCase();
        if (Generator.nameList.containsKey(name)) {
            return Generator.nameList.get(name);
        }
        return Normal.class;
    }

    public static Class getGenerator(int type) {
        if (Generator.typeList.containsKey(type)) {
            return Generator.typeList.get(type);
        }
        return Normal.class;
    }

    public static String getGeneratorName(Class c) {
        for (var entry : Generator.nameList.entrySet()) {
            if (entry.getValue().equals(c)) {
                return entry.getKey();
            }
        }
        return "unknown";
    }

    public static int getGeneratorType(Class c) {
        for (var entry : Generator.typeList.entrySet()) {
            if (entry.getValue().equals(c)) {
                return entry.getKey();
            }
        }
        return Generator.TYPE_INFINITE;
    }

    /**
     * 事实上这个方法的两个形参已经无实际意义
     */
    public abstract void init(ChunkManager chunkManager, NukkitRandom random);

    public abstract void generateChunk(int chunkX, int chunkZ);

    public abstract void populateChunk(int chunkX, int chunkZ);


    /**
     * 在指定区块上尝试生成结构
     *
     * @param chunkX
     * @param chunkZ
     */
    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public void populateStructure(int chunkX, int chunkZ) {
        //这里不能使用chunkManager而是使用level
        //因为在这个方法调用时,区块地形生成工作已完成,chunkManager(实际为PopChunkManager)内所有区块已清空
        var chunk = level.getChunk(chunkX, chunkZ);
        for (PopulatorStructure populator : structurePopulators) {
            if (populator.isAsync())
                handleAsyncStructureGenTask(new ChunkPopulationTask(level, chunk, populator));
            else populator.populate(level, chunkX, chunkZ, random, chunk);
        }
    }

    public abstract Map getSettings();

    public abstract String getName();

    public abstract Vector3 getSpawn();

    /**
     * 若返回值为true,则将会在区块地形生成完毕后调用 {@link Generator} 的 populateStructure(int chunkX, int chunkZ) 方法
     *
     * @return 是否需要生成结构
     */
    @PowerNukkitXOnly
    @Since("1.19.21-r2")
    public boolean shouldGenerateStructures() {
        return false;
    }

    /**
     * 处理需要计算的异步地形生成任务
* 有特殊需求的地形生成器可以覆写此方法并提供自己的逻辑
* 默认采用Server类的fjp线程池 * @param task 地形生成任务 */ @PowerNukkitXOnly @Since("1.19.50-r2") public void handleAsyncChunkPopTask(AsyncTask task) { //这个判断是防止单元测试报错 if (Server.getInstance().computeThreadPool != null) Server.getInstance().computeThreadPool.submit(task); } /** * 处理需要计算的异步结构生成任务
* 有特殊需求的地形生成器可以覆写此方法并提供自己的逻辑
* 默认采用Server类的fjp线程池 * @param task 结构生成任务 */ @PowerNukkitXOnly @Since("1.19.50-r2") public void handleAsyncStructureGenTask(AsyncTask task) { //这个判断是防止单元测试报错 if (Server.getInstance().computeThreadPool != null) Server.getInstance().computeThreadPool.submit(task); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy