]> git.lizzy.rs Git - BoundingBoxOutlineReloaded.git/blob - src/main/java/com/irtimaled/bbor/client/ClientRenderer.java
e4bf5b5bea5488473e23f83b84916179d9ce6fc0
[BoundingBoxOutlineReloaded.git] / src / main / java / com / irtimaled / bbor / client / ClientRenderer.java
1 package com.irtimaled.bbor.client;
2
3 import com.irtimaled.bbor.client.interop.ClientInterop;
4 import com.irtimaled.bbor.client.models.*;
5 import com.irtimaled.bbor.client.providers.*;
6 import com.irtimaled.bbor.client.renderers.*;
7 import com.irtimaled.bbor.common.MathHelper;
8 import com.irtimaled.bbor.common.TypeHelper;
9 import com.irtimaled.bbor.common.models.AbstractBoundingBox;
10 import com.irtimaled.bbor.common.models.BoundingBoxCuboid;
11 import com.irtimaled.bbor.common.models.DimensionId;
12
13 import java.util.*;
14 import java.util.stream.Stream;
15
16 public class ClientRenderer {
17     private static final int CHUNK_SIZE = 16;
18     private static final Map<Class<? extends AbstractBoundingBox>, AbstractRenderer> boundingBoxRendererMap = new HashMap<>();
19
20     private static boolean active;
21     private static final Set<IBoundingBoxProvider> providers = new HashSet<>();
22
23     public static boolean getActive() {
24         return active;
25     }
26
27     public static void toggleActive() {
28         active = !active;
29         if (!active) return;
30
31         Player.setActiveY();
32     }
33
34     static void deactivate() {
35         active = false;
36     }
37
38     static {
39         registerRenderer(BoundingBoxSlimeChunk.class, new SlimeChunkRenderer());
40         registerRenderer(BoundingBoxWorldSpawn.class, new WorldSpawnRenderer());
41         registerRenderer(BoundingBoxCuboid.class, new CuboidRenderer());
42         registerRenderer(BoundingBoxMobSpawner.class, new MobSpawnerRenderer());
43         registerRenderer(BoundingBoxSpawningSphere.class, new SpawningSphereRenderer());
44         registerRenderer(BoundingBoxBeacon.class, new BeaconRenderer());
45         registerRenderer(BoundingBoxBiomeBorder.class, new BiomeBorderRenderer());
46         registerRenderer(BoundingBoxConduit.class, new ConduitRenderer());
47         registerRenderer(BoundingBoxSpawnableBlocks.class, new SpawnableBlocksRenderer());
48         registerRenderer(BoundingBoxLine.class, new LineRenderer());
49         registerRenderer(BoundingBoxSphere.class, new SphereRenderer());
50         registerRenderer(BoundingBoxFlowerForest.class, new FlowerForestRenderer());
51
52         registerProvider(new SlimeChunkProvider());
53         registerProvider(new WorldSpawnProvider());
54         registerProvider(new SpawningSphereProvider());
55         registerProvider(new BeaconProvider());
56         registerProvider(new CustomBoxProvider());
57         registerProvider(new CustomBeaconProvider());
58         registerProvider(new BiomeBorderProvider());
59         registerProvider(new MobSpawnerProvider());
60         registerProvider(new ConduitProvider());
61         registerProvider(new SpawnableBlocksProvider());
62         registerProvider(new CustomLineProvider());
63         registerProvider(new CustomSphereProvider());
64         registerProvider(new FlowerForestProvider());
65     }
66
67     public static <T extends AbstractBoundingBox> void registerProvider(IBoundingBoxProvider<T> provider) {
68         providers.add(provider);
69     }
70
71     public static <T extends AbstractBoundingBox> void registerRenderer(Class<? extends T> type, AbstractRenderer<T> renderer) {
72         boundingBoxRendererMap.put(type, renderer);
73     }
74
75     private static boolean isWithinRenderDistance(AbstractBoundingBox boundingBox) {
76         int renderDistanceBlocks = ClientInterop.getRenderDistanceChunks() * CHUNK_SIZE;
77         int minX = MathHelper.floor(Player.getX() - renderDistanceBlocks);
78         int maxX = MathHelper.floor(Player.getX() + renderDistanceBlocks);
79         int minZ = MathHelper.floor(Player.getZ() - renderDistanceBlocks);
80         int maxZ = MathHelper.floor(Player.getZ() + renderDistanceBlocks);
81
82         return boundingBox.intersectsBounds(minX, minZ, maxX, maxZ);
83     }
84
85     public static void render(DimensionId dimensionId) {
86         if (!active) return;
87
88         RenderHelper.beforeRender();
89
90         getBoundingBoxes(dimensionId).forEach(key -> {
91             AbstractRenderer renderer = boundingBoxRendererMap.get(key.getClass());
92             if (renderer != null) renderer.render(key);
93         });
94
95         RenderHelper.afterRender();
96     }
97
98     public static Stream<AbstractBoundingBox> getBoundingBoxes(DimensionId dimensionId) {
99         Stream.Builder<AbstractBoundingBox> boundingBoxes = Stream.builder();
100         for (IBoundingBoxProvider<?> provider : providers) {
101             if (provider.canProvide(dimensionId)) {
102                 for (AbstractBoundingBox boundingBox : provider.get(dimensionId)) {
103                     if (isWithinRenderDistance(boundingBox)) {
104                         boundingBoxes.accept(boundingBox);
105                     }
106                 }
107             }
108         }
109
110         Point point = Player.getPoint();
111         return boundingBoxes.build()
112                 .sorted(Comparator
113                         .comparingDouble((AbstractBoundingBox boundingBox) -> boundingBox.getDistance(point.getX(), point.getY(), point.getZ())).reversed());
114     }
115
116     public static void clear() {
117         for(IBoundingBoxProvider<?> provider : providers) {
118             TypeHelper.doIfType(provider, ICachingProvider.class, ICachingProvider::clearCache);
119         }
120     }
121 }