package com.jozufozu.flywheel.light; import java.util.HashMap; import java.util.Map; import java.util.Set; import java.util.stream.Stream; import com.jozufozu.flywheel.util.WeakHashSet; import it.unimi.dsi.fastutil.longs.LongSet; import net.minecraft.util.math.BlockPos; import net.minecraft.util.math.SectionPos; import net.minecraft.world.IBlockDisplayReader; import net.minecraft.world.LightType; /** * Keeps track of what chunks/sections each listener is in so we can update exactly what needs to be updated. */ public class LightUpdater { private static final Map light = new HashMap<>(); public static LightUpdater get(IBlockDisplayReader world) { return light.computeIfAbsent(world, LightUpdater::new); } private final LightProvider provider; private final WeakHashSet movingListeners = new WeakHashSet<>(); private final WeakContainmentMultiMap sections = new WeakContainmentMultiMap<>(); private final WeakContainmentMultiMap chunks = new WeakContainmentMultiMap<>(); public LightUpdater(IBlockDisplayReader world) { provider = BasicProvider.get(world); } public void tick() { for (IMovingListener listener : movingListeners) { if (listener.update(provider)) { addListener(listener); } } } /** * Add a listener. * @param listener The object that wants to receive light update notifications. */ public void addListener(ILightUpdateListener listener) { if (listener instanceof IMovingListener) movingListeners.add(((IMovingListener) listener)); GridAlignedBB box = listener.getVolume(); LongSet sections = this.sections.getAndResetContainment(listener); LongSet chunks = this.chunks.getAndResetContainment(listener); int minX = SectionPos.blockToSectionCoord(box.minX); int minY = SectionPos.blockToSectionCoord(box.minY); int minZ = SectionPos.blockToSectionCoord(box.minZ); int maxX = SectionPos.blockToSectionCoord(box.maxX); int maxY = SectionPos.blockToSectionCoord(box.maxY); int maxZ = SectionPos.blockToSectionCoord(box.maxZ); for (int x = minX; x <= maxX; x++) { for (int z = minZ; z <= maxZ; z++) { for (int y = minY; y <= maxY; y++) { long sectionPos = SectionPos.asLong(x, y, z); this.sections.put(sectionPos, listener); sections.add(sectionPos); } long chunkPos = SectionPos.asLong(x, 0, z); this.chunks.put(chunkPos, listener); chunks.add(chunkPos); } } } /** * Dispatch light updates to all registered {@link ILightUpdateListener}s. * @param type The type of light that changed. * @param sectionPos A long representing the section position where light changed. */ public void onLightUpdate(LightType type, long sectionPos) { Set set = sections.get(sectionPos); if (set == null || set.isEmpty()) return; set.removeIf(l -> l.status().shouldRemove()); GridAlignedBB chunkBox = GridAlignedBB.from(SectionPos.of(sectionPos)); for (ILightUpdateListener listener : set) { listener.onLightUpdate(provider, type, chunkBox.copy()); } } /** * Dispatch light updates to all registered {@link ILightUpdateListener}s * when the server sends lighting data for an entire chunk. * */ public void onLightPacket(int chunkX, int chunkZ) { long chunkPos = SectionPos.asLong(chunkX, 0, chunkZ); Set set = chunks.get(chunkPos); if (set == null || set.isEmpty()) return; set.removeIf(l -> l.status().shouldRemove()); for (ILightUpdateListener listener : set) { listener.onLightPacket(provider, chunkX, chunkZ); } } public static long blockToSection(BlockPos pos) { return SectionPos.asLong(pos.getX(), pos.getY(), pos.getZ()); } public static long sectionToChunk(long sectionPos) { return sectionPos & 0xFFFFFFFFFFF_00000L; } public Stream getAllBoxes() { return chunks.stream().map(ILightUpdateListener::getVolume); } public boolean isEmpty() { return chunks.isEmpty(); } }