diff --git a/CHANGELOG.md b/CHANGELOG.md index 992f809c..bc341862 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,3 +1,29 @@ +> ### DynamicSurroundings-1.21.1-0.4.2 +**All Loaders** +* JAVA 21+ +* Architectury 13.0.8+ + +**Fabric** +* Fabric Loader >= 0.16.9 +* Fabric API >= 0.110.0+1.21. + +**NeoForge** +* NeoForge 21.1.84+ + +**What's New** +* Experimental: Toolbar sounds will play the "block step" sound for block items. It is off by default, and can be turned on in settings. +* Added whatsplaying subcommand to /dsmm. Will report on the music that is playing in the Music Manager. +* Work in Progress: Brought back a variety of block step sounds. Dynamic Surroundings will perform remapping of sound plays to get an updated sound, which is entirely different from the prior implementation. This is not intended to be full-featured as with prior step simulations. Can be disabled in the configuration (Sound Options -> Sound Remapping). + +**Changes** +* Do not tint biome fog if biome fog effect is disabled. +* Compass overlay will spin wildly if the dimension is not natural, like the Nether. This mirrors vanilla compass behavior. +* Internal modifications and restructure to facilitate porting to MC 1.21.4. Mojang started its refactor for how component data is encoded, not to mention the always welcome "find out what was renamed and where it was moved to" game. + +**Fixes** +* Disabling fog effect actually works +* Compatibility with Nostalgia Tweaks and Distant Horizons world fog effect + > ### DynamicSurroundings-1.21.1-0.4.1 **All Loaders** * JAVA 21+ diff --git a/build.gradle b/build.gradle index 0114ec28..88e8d892 100644 --- a/build.gradle +++ b/build.gradle @@ -58,6 +58,14 @@ subprojects { it.options.release = 21 } + tasks.register('copyJarToMain', Copy) { + duplicatesStrategy = DuplicatesStrategy.INCLUDE + if (project.name != "common" && project.name != "stubs") { + from remapJar + into "../build/libs" + } + } + // Configure Maven publishing. publishing { publications { diff --git a/common/src/main/java/org/orecruncher/dsurround/Client.java b/common/src/main/java/org/orecruncher/dsurround/Client.java index d71bd8f2..a2505a11 100644 --- a/common/src/main/java/org/orecruncher/dsurround/Client.java +++ b/common/src/main/java/org/orecruncher/dsurround/Client.java @@ -9,7 +9,6 @@ import org.orecruncher.dsurround.commands.Commands; import org.orecruncher.dsurround.config.libraries.*; import org.orecruncher.dsurround.config.libraries.impl.*; -import org.orecruncher.dsurround.effects.particles.ParticleSheets; import org.orecruncher.dsurround.gui.overlay.OverlayManager; import org.orecruncher.dsurround.gui.keyboard.KeyBindings; import org.orecruncher.dsurround.lib.GameUtils; @@ -24,6 +23,8 @@ import org.orecruncher.dsurround.eventing.ClientState; import org.orecruncher.dsurround.lib.registry.ReloadListener; import org.orecruncher.dsurround.lib.resources.ResourceUtilities; +import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; +import org.orecruncher.dsurround.lib.seasons.SeasonManager; import org.orecruncher.dsurround.lib.version.IVersionChecker; import org.orecruncher.dsurround.lib.version.VersionChecker; import org.orecruncher.dsurround.lib.version.VersionResult; @@ -128,6 +129,8 @@ public void initializeClient() { .registerSingleton(ISoundLibrary.class, SoundLibrary.class) .registerSingleton(IBiomeLibrary.class, BiomeLibrary.class) .registerSingleton(IDimensionLibrary.class, DimensionLibrary.class) + .registerSingleton(IDimensionInformation.class, DimensionInformation.class) + .registerFactory(ISeasonalInformation.class, () -> SeasonManager.HANDLER) .registerSingleton(IBlockLibrary.class, BlockLibrary.class) .registerSingleton(IItemLibrary.class, ItemLibrary.class) .registerSingleton(IEntityEffectLibrary.class, EntityEffectLibrary.class) @@ -180,11 +183,6 @@ public void onComplete(Minecraft client) { // of the dependencies to be initialized. container.resolve(Handlers.class); - // Make sure our particle sheets get registered otherwise they will not render. - // These sheets are purely client side - they have to be manhandled into the - // Minecraft environment. - ParticleSheets.register(); - this.logger.info("[%s] Finalization complete", Constants.MOD_ID); } diff --git a/common/src/main/java/org/orecruncher/dsurround/Configuration.java b/common/src/main/java/org/orecruncher/dsurround/Configuration.java index f8cf16cc..ec664c40 100644 --- a/common/src/main/java/org/orecruncher/dsurround/Configuration.java +++ b/common/src/main/java/org/orecruncher/dsurround/Configuration.java @@ -165,6 +165,10 @@ public static class SoundOptions { @Property @Comment("Enables display of toast messages for credited music") public boolean displayToastMessagesForMusic = true; + + @Property + @Comment("Enables sound remapping when sounds are played") + public boolean remapSounds = true; } public static class BlockEffects { @@ -232,6 +236,11 @@ public static class EntityEffects { @Comment("Enable/disable player toolbar sound effects") public boolean enablePlayerToolbarEffect = true; + @Property + @RestartRequired(client = false) + @Comment("Enable/disable sound effects for blocks on the toolbar") + public boolean enableToolbarBlockSounds = false; + @Property @RestartRequired(client = false) @Comment("Enable/disable item swing sound effects from players and mobs") @@ -259,10 +268,6 @@ public static class FootstepAccents { @Property @Comment("Enable/disable accents when the player is walking on squeaky blocks") public boolean enableFloorSqueaks = true; - - @Property - @Comment("Enable/disable accents for when the player is walking on leafy blocks") - public boolean enableLeafAccents = true; } public static class ParticleTweaks { diff --git a/common/src/main/java/org/orecruncher/dsurround/commands/MusicManagerCommand.java b/common/src/main/java/org/orecruncher/dsurround/commands/MusicManagerCommand.java index e8f7ff43..12cc8f68 100644 --- a/common/src/main/java/org/orecruncher/dsurround/commands/MusicManagerCommand.java +++ b/common/src/main/java/org/orecruncher/dsurround/commands/MusicManagerCommand.java @@ -10,12 +10,14 @@ public class MusicManagerCommand extends AbstractClientCommand { private static final String RESET = "reset"; private static final String PAUSE = "pause"; private static final String UNPAUSE = "unpause"; + private static final String WHATS_PLAYING = "whatsplaying"; @Override public void register(CommandDispatcher dispatcher, CommandBuildContext registryAccess) { dispatcher.register(ClientCommandRegistrationEvent.literal(COMMAND) .then(subCommand(RESET, MusicManagerCommandHandler::reset)) .then(subCommand(PAUSE, MusicManagerCommandHandler::pause)) - .then(subCommand(UNPAUSE, MusicManagerCommandHandler::unpause))); + .then(subCommand(UNPAUSE, MusicManagerCommandHandler::unpause)) + .then(subCommand(WHATS_PLAYING, MusicManagerCommandHandler::whatsPlaying))); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/commands/handlers/MusicManagerCommandHandler.java b/common/src/main/java/org/orecruncher/dsurround/commands/handlers/MusicManagerCommandHandler.java index bf7f7f17..3888769c 100644 --- a/common/src/main/java/org/orecruncher/dsurround/commands/handlers/MusicManagerCommandHandler.java +++ b/common/src/main/java/org/orecruncher/dsurround/commands/handlers/MusicManagerCommandHandler.java @@ -2,6 +2,7 @@ import net.minecraft.network.chat.Component; import org.orecruncher.dsurround.lib.GameUtils; +import org.orecruncher.dsurround.lib.config.ConfigurationData; import org.orecruncher.dsurround.mixinutils.IMusicManager; public class MusicManagerCommandHandler { @@ -32,4 +33,13 @@ public static Component pause() { return Component.translatable("dsurround.command.dsmm.pause.failure", t.getMessage()); } } + + public static Component whatsPlaying() { + try { + var result = ((IMusicManager)(GameUtils.getMC().getMusicManager())).dsurround_whatsPlaying(); + return Component.translatable("dsurround.command.dsmm.whatsplaying.success", result); + } catch (Throwable t) { + return Component.translatable("dsurround.command.dsmm.whatsplaying.failure", t.getMessage()); + } + } } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntry.java b/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntry.java index 61018fac..5a043a8a 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntry.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntry.java @@ -1,6 +1,8 @@ package org.orecruncher.dsurround.config; import com.google.common.base.MoreObjects; +import net.minecraft.util.random.Weight; +import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.orecruncher.dsurround.lib.WeightTable; import org.orecruncher.dsurround.lib.di.ContainerManager; @@ -11,9 +13,9 @@ public class AcousticEntry implements WeightTable.IItem { private static final IConditionEvaluator CONDITION_EVALUATOR = ContainerManager.resolve(IConditionEvaluator.class); - private static final int DEFAULT_WEIGHT = 10; + private static final Weight DEFAULT_WEIGHT = Weight.of(10); - private final int weight; + private final Weight weight; private final ISoundFactory acoustic; private final Script conditions; @@ -21,19 +23,20 @@ public AcousticEntry(final ISoundFactory acoustic, @Nullable final Script condit this(acoustic, condition, DEFAULT_WEIGHT); } - public AcousticEntry(final ISoundFactory acoustic, @Nullable final Script condition, int weight) { + public AcousticEntry(final ISoundFactory acoustic, @Nullable final Script condition, Weight weight) { this.acoustic = acoustic; this.weight = weight; this.conditions = condition != null ? condition : Script.TRUE; } + @NotNull @Override - public int getWeight() { + public Weight getWeight() { return this.weight; } @Override - public ISoundFactory getItem() { + public ISoundFactory data() { return getAcoustic(); } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntryCollection.java b/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntryCollection.java index ae84912c..99aca4ec 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntryCollection.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/AcousticEntryCollection.java @@ -1,14 +1,71 @@ package org.orecruncher.dsurround.config; +import net.minecraft.util.random.SimpleWeightedRandomList; +import org.orecruncher.dsurround.lib.WeightTable; import org.orecruncher.dsurround.lib.collections.ObjectArray; +import org.orecruncher.dsurround.lib.random.Randomizer; +import org.orecruncher.dsurround.sound.ISoundFactory; +import java.util.Optional; +import java.util.stream.Stream; + +@SuppressWarnings("unused") public class AcousticEntryCollection extends ObjectArray { + public static final AcousticEntryCollection EMPTY; + + static { + EMPTY = new AcousticEntryCollection() { + @Override + public boolean add(AcousticEntry entry) { + throw new RuntimeException("Cannot add AcousticEntry to EMPTY collection"); + } + @Override + public Stream findMatches() { + return Stream.empty(); + } + @Override + public Optional makeSelection() { + return Optional.empty(); + } + @Override + public SimpleWeightedRandomList matchesAsWeightedList() { + return SimpleWeightedRandomList.empty(); + } + }; + EMPTY.trim(); + } + @Override public boolean add(AcousticEntry entry) { if (this.contains(entry)) return false; - return super.add(entry); } + + /** + * Stream of AcousticEntries that match the current conditions within + * the game. + */ + public Stream findMatches() { + return this.stream().filter(AcousticEntry::matches); + } + + /** + * Creates a SimpleWeightedRandomList based on valid candidates from within + * the collection. + */ + public SimpleWeightedRandomList matchesAsWeightedList() { + var builder = new SimpleWeightedRandomList.Builder(); + this.findMatches().forEach(m -> builder.add(m.getAcoustic(), m.getWeight().asInt())); + return builder.build(); + } + + /** + * Makes a weighted choice from the candidates available in the + * collection. + */ + public Optional makeSelection() { + return WeightTable.makeSelection(this.findMatches(), Randomizer.current()); + } } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/dimension/DimensionInfo.java b/common/src/main/java/org/orecruncher/dsurround/config/DimensionInfo.java similarity index 89% rename from common/src/main/java/org/orecruncher/dsurround/config/dimension/DimensionInfo.java rename to common/src/main/java/org/orecruncher/dsurround/config/DimensionInfo.java index ef2bc045..33d85bd9 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/dimension/DimensionInfo.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/DimensionInfo.java @@ -1,4 +1,4 @@ -package org.orecruncher.dsurround.config.dimension; +package org.orecruncher.dsurround.config; import net.minecraft.resources.ResourceLocation; import net.minecraft.world.level.Level; @@ -19,6 +19,7 @@ public class DimensionInfo { protected int spaceHeight; protected boolean alwaysOutside = false; protected boolean playBiomeSounds = true; + protected boolean compassWobble = false; DimensionInfo() { this.name = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "no_dimension"); @@ -37,6 +38,8 @@ public DimensionInfo(final Level world) { // Force sea level based on known world types that give heartburn if (this.isFlatWorld) this.seaLevel = 0; + + this.compassWobble = !world.dimensionType().natural(); } public void update(DimensionConfigRule config) { @@ -49,6 +52,8 @@ public void update(DimensionConfigRule config) { v -> this.cloudHeight = v, () -> this.cloudHeight = this.skyHeight / 2); + config.compassWobble().ifPresent(v -> this.compassWobble = v); + this.spaceHeight = this.skyHeight + SPACE_HEIGHT_OFFSET; } } @@ -85,4 +90,8 @@ public boolean isFlatWorld() { return this.isFlatWorld; } + public boolean getCompassWobble() { + return this.compassWobble; + } + } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/SoundMapping.java b/common/src/main/java/org/orecruncher/dsurround/config/SoundMapping.java new file mode 100644 index 00000000..9a9dba68 --- /dev/null +++ b/common/src/main/java/org/orecruncher/dsurround/config/SoundMapping.java @@ -0,0 +1,109 @@ +package org.orecruncher.dsurround.config; + +import net.minecraft.resources.ResourceLocation; +import net.minecraft.world.level.block.state.BlockState; +import org.jetbrains.annotations.Nullable; +import org.orecruncher.dsurround.config.data.SoundMappingConfigRule; +import org.orecruncher.dsurround.lib.IMatcher; +import org.orecruncher.dsurround.lib.collections.ObjectArray; + +import java.util.Optional; + +public record SoundMapping(ResourceLocation soundEvent, ObjectArray rules) { + + public static SoundMapping of(SoundMappingConfigRule rule) { + ObjectArray mappings = new ObjectArray<>(rule.rules().size()); + rule.rules().forEach(r -> mappings.add(Mapping.of(r))); + return new SoundMapping(rule.soundEvent(), mappings); + } + + public boolean isBlockStateNeeded() { + return !this.rules.isEmpty() && !this.rules.getFirst().isDefaultRule(); + } + + public Optional findMatch(@Nullable BlockState state) { + Optional factory = Optional.empty(); + for (var rule : this.rules) { + factory = rule.findMatch(state); + if (factory.isPresent()) + break; + } + return factory; + } + + public void merge(SoundMappingConfigRule mapping) { + if (!this.soundEvent.equals(mapping.soundEvent())) + throw new RuntimeException("Unable to merge sound mapping rule - factories do not match"); + + for (var rule : mapping.rules()) { + var mapped = Mapping.of(rule); + + // Find the first applicable rule that matches the factory that is needed. + // It's possible to have multiple rules with the same factory. It can occur when + // merging two different rule definitions where one has the factory as a default + // and the other has block matchers. + var existingRule = this.rules.stream().filter(r -> r.factory().equals(rule.factory())).findFirst(); + if (existingRule.isPresent()) { + // If it is a default rule, we do not want to merge. Instead, we + // insert prior. We need to preserve the existing rule as default. + if (existingRule.get().isDefaultRule()) { + this.insertBeforeDefaultRule(mapped); + } else { + // Need to add block matcher definitions + existingRule.get().merge(mapped); + } + } else { + // Need to add the rule. If the last rule in the collection is all matches, we need + // to insert prior. Otherwise, we append. + var last = this.rules.getLast(); + if (last.isDefaultRule()) { + this.insertBeforeDefaultRule(mapped); + } else { + this.rules.add(mapped); + } + } + } + } + + private void insertBeforeDefaultRule(Mapping mapping) { + var last = this.rules.getLast(); + if (!last.isDefaultRule()) + throw new RuntimeException("Last rule in sound mapping configuration is not default"); + this.rules.remove(last); + this.rules.add(mapping); + this.rules.add(last); + } + + public record Mapping(ObjectArray> blocks, ResourceLocation factory) { + + public static Mapping of(SoundMappingConfigRule.MappingRule mappingRule) { + ObjectArray> blocks = new ObjectArray<>(mappingRule.blocks().size()); + blocks.addAll(mappingRule.blocks()); + return new Mapping(blocks, mappingRule.factory()); + } + + public Optional findMatch(@Nullable BlockState state) { + if (this.isDefaultRule()) + return Optional.of(this.factory); + // Since the rules have BlockState matching if a null state is provided + // return empty - nothing could be matched. + if (state == null) + return Optional.empty(); + for( var rule : this.blocks) { + if (rule.match(state)) + return Optional.of(this.factory); + } + return Optional.empty(); + } + + public boolean isDefaultRule() { + return this.blocks.isEmpty(); + } + + public void merge(Mapping rule) { + if (!this.factory.equals(rule.factory())) + throw new RuntimeException("Unable to add mapping rule - factories do not match"); + this.blocks.addAll(rule.blocks()); + } + } +} diff --git a/common/src/main/java/org/orecruncher/dsurround/config/WaterRippleStyle.java b/common/src/main/java/org/orecruncher/dsurround/config/WaterRippleStyle.java index 343ae4c6..f2e3cdc4 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/WaterRippleStyle.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/WaterRippleStyle.java @@ -2,13 +2,12 @@ import net.minecraft.resources.ResourceLocation; import org.orecruncher.dsurround.Constants; -import org.orecruncher.dsurround.effects.particles.ParticleSheets; import org.orecruncher.dsurround.lib.random.Randomizer; public enum WaterRippleStyle { - NONE (ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "none")), - PIXELATED_CIRCLE(ParticleSheets.TEXTURE_WATER_RIPPLE_PIXELATED_CIRCLE) { + NONE("none"), + PIXELATED_CIRCLE("textures/particles/pixel_ripples.png") { private final int FRAMES = 7; private final float DELTA = 1F / this.FRAMES; private final int MAX_AGE = this.FRAMES * 2; @@ -36,8 +35,8 @@ public int getMaxAge() { private final ResourceLocation resource; - WaterRippleStyle(final ResourceLocation texture) { - this.resource = texture; + WaterRippleStyle(final String texture) { + this.resource = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, texture); } public ResourceLocation getTexture() { diff --git a/common/src/main/java/org/orecruncher/dsurround/config/biome/BiomeInfo.java b/common/src/main/java/org/orecruncher/dsurround/config/biome/BiomeInfo.java index 22862de6..f3a93f50 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/biome/BiomeInfo.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/biome/BiomeInfo.java @@ -5,6 +5,7 @@ import net.minecraft.network.chat.TextColor; import net.minecraft.resources.ResourceLocation; import net.minecraft.sounds.Music; +import net.minecraft.util.random.Weight; import net.minecraft.world.level.biome.Biome; import org.apache.commons.lang3.StringUtils; import org.jetbrains.annotations.Nullable; @@ -20,7 +21,6 @@ import org.orecruncher.dsurround.lib.logging.ModLog; import org.orecruncher.dsurround.lib.random.IRandomizer; import org.orecruncher.dsurround.lib.registry.RegistryUtils; -import org.orecruncher.dsurround.lib.WeightTable; import org.orecruncher.dsurround.lib.collections.ObjectArray; import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.lib.logging.IModLog; @@ -51,10 +51,10 @@ public final class BiomeInfo implements Comparable, IBiomeSoundProvid private final boolean isOcean; private final boolean isDeepOcean; private final boolean isCave; - private Collection loopSounds = new AcousticEntryCollection(); - private Collection moodSounds = new AcousticEntryCollection(); - private Collection additionalSounds = new AcousticEntryCollection(); - private Collection musicSounds = new AcousticEntryCollection(); + private AcousticEntryCollection loopSounds = new AcousticEntryCollection(); + private AcousticEntryCollection moodSounds = new AcousticEntryCollection(); + private AcousticEntryCollection additionalSounds = new AcousticEntryCollection(); + private AcousticEntryCollection musicSounds = new AcousticEntryCollection(); private Collection comments = new ObjectArray<>(); private TextColor fogColor; private FogDensity fogDensity; @@ -165,8 +165,8 @@ public boolean hasTrait(String trait) { @Override public Collection findBiomeSoundMatches() { - return this.loopSounds.stream() - .filter(AcousticEntry::matches) + return this.loopSounds + .findMatches() .map(AcousticEntry::getAcoustic) .collect(Collectors.toList()); } @@ -175,7 +175,7 @@ public Collection findBiomeSoundMatches() { public Optional getExtraSound(final SoundEventType type, final IRandomizer random) { @Nullable - Collection sourceList = null; + AcousticEntryCollection sourceList = null; switch (type) { case ADDITION -> { @@ -193,11 +193,7 @@ public Optional getExtraSound(final SoundEventType type, final IR case MUSIC -> sourceList = this.musicSounds; } - if (sourceList == null || sourceList.isEmpty()) - return Optional.empty(); - - var candidates = sourceList.stream().filter(AcousticEntry::matches); - return WeightTable.makeSelection(candidates); + return sourceList == null ? Optional.empty() : sourceList.makeSelection(); } @Override @@ -242,7 +238,7 @@ public void update(final BiomeConfigRule entry) { targetCollection = this.loopSounds; } case MUSIC, MOOD, ADDITION -> { - final int weight = sr.weight(); + final Weight weight = sr.weight(); acousticEntry = new AcousticEntry(factory, sr.conditions(), weight); if (sr.type() == SoundEventType.ADDITION) @@ -265,13 +261,13 @@ else if (sr.type() == SoundEventType.MOOD) public void trim() { if (this.loopSounds.isEmpty()) - this.loopSounds = ImmutableList.of(); + this.loopSounds = AcousticEntryCollection.EMPTY; if (this.moodSounds.isEmpty()) - this.moodSounds = ImmutableList.of(); + this.moodSounds = AcousticEntryCollection.EMPTY; if (this.additionalSounds.isEmpty()) - this.additionalSounds = ImmutableList.of(); + this.additionalSounds = AcousticEntryCollection.EMPTY; if (this.musicSounds.isEmpty()) - this.musicSounds = ImmutableList.of(); + this.musicSounds = AcousticEntryCollection.EMPTY; if (this.comments.isEmpty()) this.comments = ImmutableList.of(); } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/block/BlockInfo.java b/common/src/main/java/org/orecruncher/dsurround/config/block/BlockInfo.java index 659a1b32..ab75b209 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/block/BlockInfo.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/block/BlockInfo.java @@ -1,8 +1,10 @@ package org.orecruncher.dsurround.config.block; import com.google.common.collect.ImmutableList; +import net.minecraft.resources.ResourceLocation; import net.minecraft.tags.BlockTags; import net.minecraft.world.level.block.state.BlockState; +import org.jetbrains.annotations.Nullable; import org.orecruncher.dsurround.config.AcousticEntryCollection; import org.orecruncher.dsurround.config.data.AcousticConfig; import org.orecruncher.dsurround.config.libraries.ISoundLibrary; @@ -10,7 +12,6 @@ import org.orecruncher.dsurround.config.data.BlockConfigRule; import org.orecruncher.dsurround.config.libraries.ITagLibrary; import org.orecruncher.dsurround.effects.IBlockEffectProducer; -import org.orecruncher.dsurround.lib.WeightTable; import org.orecruncher.dsurround.lib.collections.ObjectArray; import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.lib.logging.IModLog; @@ -60,7 +61,9 @@ private static class Reflectance { private static final ITagLibrary TAG_LIBRARY = ContainerManager.resolve(ITagLibrary.class); protected final int version; - protected Collection sounds = new AcousticEntryCollection(); + @Nullable + protected final ResourceLocation stepSound; + protected AcousticEntryCollection sounds = new AcousticEntryCollection(); protected Collection blockEffects = new ObjectArray<>(); protected Script soundChance = new Script("0.01"); @@ -69,12 +72,14 @@ private static class Reflectance { public BlockInfo(int version) { this.version = version; + this.stepSound = null; } public BlockInfo(int version, BlockState state) { this.version = version; this.soundOcclusion = getSoundOcclusionSetting(state); this.soundReflectivity = getSoundReflectionSetting(state); + this.stepSound = state.getSoundType().getStepSound().getLocation(); } public boolean isDefault() { @@ -100,7 +105,6 @@ private void addToBlockEffects(IBlockEffectProducer effect) { this.blockEffects.add(effect); } - // TODO: Eliminate duplicates public void update(BlockConfigRule config) { // Reset of a block clears all registries if (config.clearSounds()) @@ -134,8 +138,7 @@ public Optional getSoundToPlay(final IRandomizer random) { if (this.sounds != null) { var chance = CONDITION_EVALUATOR.eval(this.soundChance); if (chance instanceof Double c && random.nextDouble() < c) { - var candidates = this.sounds.stream().filter(AcousticEntry::matches); - return WeightTable.makeSelection(candidates); + return this.sounds.makeSelection(); } } return Optional.empty(); @@ -147,7 +150,7 @@ public Collection getEffectProducers() { public void trim() { if (this.sounds.isEmpty()) { - this.sounds = ImmutableList.of(); + this.sounds = AcousticEntryCollection.EMPTY; } if (this.blockEffects.isEmpty()) { this.blockEffects = ImmutableList.of(); @@ -333,6 +336,10 @@ public String toString() { .append(this.soundOcclusion) .append("\n"); + if (this.stepSound != null) { + builder.append("step sound: ").append(this.stepSound).append("\n"); + } + if (!this.sounds.isEmpty()) { builder.append("sound chance: ").append(this.soundChance); builder.append("; sounds [\n"); diff --git a/common/src/main/java/org/orecruncher/dsurround/config/data/AcousticConfig.java b/common/src/main/java/org/orecruncher/dsurround/config/data/AcousticConfig.java index db8149b9..bb9abae2 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/data/AcousticConfig.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/data/AcousticConfig.java @@ -3,6 +3,7 @@ import com.mojang.serialization.Codec; import com.mojang.serialization.codecs.RecordCodecBuilder; import net.minecraft.resources.ResourceLocation; +import net.minecraft.util.random.Weight; import org.orecruncher.dsurround.config.SoundEventType; import org.orecruncher.dsurround.lib.IdentityUtils; import org.orecruncher.dsurround.lib.scripting.Script; @@ -10,14 +11,16 @@ public record AcousticConfig( ResourceLocation factory, Script conditions, - Integer weight, + Weight weight, SoundEventType type) { + private static final Weight DEFAULT_WEIGHT = Weight.of(10); + public static final Codec CODEC = RecordCodecBuilder.create((instance) -> instance.group( IdentityUtils.CODEC.fieldOf("factory").forGetter(AcousticConfig::factory), Script.CODEC.optionalFieldOf("conditions", Script.TRUE).forGetter(AcousticConfig::conditions), - Codec.intRange(0, Integer.MAX_VALUE).optionalFieldOf("weight", 10).forGetter(AcousticConfig::weight), + Weight.CODEC.optionalFieldOf("weight", DEFAULT_WEIGHT).forGetter(AcousticConfig::weight), SoundEventType.CODEC.optionalFieldOf("type", SoundEventType.LOOP).forGetter(AcousticConfig::type) ).apply(instance, AcousticConfig::new)); } \ No newline at end of file diff --git a/common/src/main/java/org/orecruncher/dsurround/config/data/DimensionConfigRule.java b/common/src/main/java/org/orecruncher/dsurround/config/data/DimensionConfigRule.java index ff5284c6..b33bfa3c 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/data/DimensionConfigRule.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/data/DimensionConfigRule.java @@ -12,7 +12,8 @@ public record DimensionConfigRule( Optional skyHeight, Optional cloudHeight, Optional alwaysOutside, - Optional playBiomeSounds) { + Optional playBiomeSounds, + Optional compassWobble) { public static final Codec CODEC = RecordCodecBuilder.create((instance) -> instance.group( ResourceLocation.CODEC.fieldOf("dimId").forGetter(DimensionConfigRule::dimensionId), @@ -20,7 +21,8 @@ public record DimensionConfigRule( Codec.INT.optionalFieldOf("skyHeight").forGetter(DimensionConfigRule::skyHeight), Codec.INT.optionalFieldOf("cloudHeight").forGetter(DimensionConfigRule::cloudHeight), Codec.BOOL.optionalFieldOf("alwaysOutside").forGetter(DimensionConfigRule::alwaysOutside), - Codec.BOOL.optionalFieldOf("playBiomeSounds").forGetter(DimensionConfigRule::playBiomeSounds)) + Codec.BOOL.optionalFieldOf("playBiomeSounds").forGetter(DimensionConfigRule::playBiomeSounds), + Codec.BOOL.optionalFieldOf("compassWobble").forGetter(DimensionConfigRule::compassWobble)) .apply(instance, DimensionConfigRule::new)); @Override @@ -32,6 +34,7 @@ public String toString() { this.cloudHeight.ifPresent(v -> builder.append(" cloudHeight: ").append(v)); this.alwaysOutside.ifPresent(v -> builder.append(" alwaysOutside: ").append(v)); this.playBiomeSounds.ifPresent(v -> builder.append(" playBiomeSounds: ").append(v)); + this.compassWobble.ifPresent(v -> builder.append(" compassWobble: ").append(v)); return builder.toString(); } diff --git a/common/src/main/java/org/orecruncher/dsurround/config/data/SoundMappingConfigRule.java b/common/src/main/java/org/orecruncher/dsurround/config/data/SoundMappingConfigRule.java new file mode 100644 index 00000000..b712fd7e --- /dev/null +++ b/common/src/main/java/org/orecruncher/dsurround/config/data/SoundMappingConfigRule.java @@ -0,0 +1,34 @@ +package org.orecruncher.dsurround.config.data; + +import com.google.common.collect.ImmutableList; +import com.mojang.serialization.Codec; +import com.mojang.serialization.codecs.RecordCodecBuilder; +import net.minecraft.resources.ResourceLocation; +import net.minecraft.world.level.block.state.BlockState; +import org.orecruncher.dsurround.lib.CodecExtensions; +import org.orecruncher.dsurround.lib.IMatcher; +import org.orecruncher.dsurround.lib.IdentityUtils; + +import java.util.List; + +/** + * Sound mapping configuration rule. The order of the rules is important as they are processed sequentially. For + * mappings that have more than one rule, the default is placed as the last entry without any BlockState + * specifications. + */ +public record SoundMappingConfigRule(ResourceLocation soundEvent, List rules) { + + public static final Codec CODEC = RecordCodecBuilder.create((instance) -> + instance.group( + IdentityUtils.CODEC.fieldOf("soundEvent").forGetter(SoundMappingConfigRule::soundEvent), + Codec.list(MappingRule.CODEC).fieldOf("rules").forGetter(SoundMappingConfigRule::rules) + ).apply(instance, SoundMappingConfigRule::new)); + + public record MappingRule(List> blocks, ResourceLocation factory) { + + public static final Codec CODEC = RecordCodecBuilder.create((instance) -> + instance.group( + Codec.list(CodecExtensions.checkBlockStateSpecification(true)).optionalFieldOf("blocks", ImmutableList.of()).forGetter(MappingRule::blocks), + IdentityUtils.CODEC.fieldOf("factory").forGetter(MappingRule::factory)).apply(instance, MappingRule::new)); + } +} diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionInformation.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionInformation.java new file mode 100644 index 00000000..389f1e8d --- /dev/null +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionInformation.java @@ -0,0 +1,36 @@ +package org.orecruncher.dsurround.config.libraries; + +import net.minecraft.client.multiplayer.ClientLevel; +import net.minecraft.resources.ResourceLocation; + +public interface IDimensionInformation { + /** + * The resource location ID of the dimension + */ + ResourceLocation name(); + /** + * The client level for the dimension + */ + ClientLevel level(); + /** + * The sea level configured for the dimension + */ + int seaLevel(); + /** + * Whether the dimension is considered always outside, like Nether. + */ + boolean alwaysOutside(); + /** + * The vertical Y level which is the threshold of outer space. + */ + int getSpaceHeight(); + /** + * The veritical Y level where clouds are expected to be + */ + int getCloudHeight(); + + /** + * Indicates whether the compass should "wobble" making the bearing unreadable + */ + boolean getCompassWobble(); +} diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionLibrary.java index ff68b27c..424a118f 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/IDimensionLibrary.java @@ -1,7 +1,7 @@ package org.orecruncher.dsurround.config.libraries; import net.minecraft.world.level.Level; -import org.orecruncher.dsurround.config.dimension.DimensionInfo; +import org.orecruncher.dsurround.config.DimensionInfo; public interface IDimensionLibrary extends ILibrary { DimensionInfo getData(final Level world); diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/ISoundLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/ISoundLibrary.java index 4177c932..974bcf11 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/ISoundLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/ISoundLibrary.java @@ -1,5 +1,6 @@ package org.orecruncher.dsurround.config.libraries; +import net.minecraft.client.resources.sounds.SoundInstance; import net.minecraft.resources.ResourceLocation; import net.minecraft.sounds.Music; import net.minecraft.sounds.SoundEvent; @@ -21,6 +22,7 @@ public interface ISoundLibrary extends ILibrary { ISoundFactory getSoundFactoryOrDefault(ResourceLocation factoryLocation); ISoundFactory getSoundFactoryForMusic(Music music); + Optional remapSound(SoundInstance soundInstance); boolean isBlocked(final ResourceLocation id); boolean isCulled(final ResourceLocation id); float getVolumeScale(SoundSource category, ResourceLocation id); diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/BiomeLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/BiomeLibrary.java index 6274e86c..239c9c9b 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/BiomeLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/BiomeLibrary.java @@ -42,7 +42,7 @@ public final class BiomeLibrary implements IBiomeLibrary { // Cached list of biome config rules. Need to hold onto them // because they may be needed to handle a dynamic biome load. - private final ObjectArray biomeConfigs = new ObjectArray<>(64); + private final ObjectArray biomeConfigs = new ObjectArray<>(128); // Current version of the configs that are loaded. Used to detect when // configs changed and cached biome info needs a refresh. diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionInformation.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionInformation.java new file mode 100644 index 00000000..e89c0c4f --- /dev/null +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionInformation.java @@ -0,0 +1,60 @@ +package org.orecruncher.dsurround.config.libraries.impl; + +import dev.architectury.event.events.client.ClientLifecycleEvent; +import net.minecraft.client.multiplayer.ClientLevel; +import net.minecraft.resources.ResourceLocation; +import org.orecruncher.dsurround.config.DimensionInfo; +import org.orecruncher.dsurround.config.libraries.AssetLibraryEvent; +import org.orecruncher.dsurround.config.libraries.IDimensionInformation; +import org.orecruncher.dsurround.config.libraries.IDimensionLibrary; +import org.orecruncher.dsurround.lib.GameUtils; +import org.orecruncher.dsurround.lib.events.HandlerPriority; + +public class DimensionInformation implements IDimensionInformation { + + private final IDimensionLibrary dimensionLibrary; + private DimensionInfo info; + + public DimensionInformation(IDimensionLibrary dimensionLibrary) { + this.dimensionLibrary = dimensionLibrary; + + // Need to reset the cached dimension info whenever the client world + // changes or if there is a resource reload. + ClientLifecycleEvent.CLIENT_LEVEL_LOAD.register(state -> this.info = null); + AssetLibraryEvent.RELOAD.register((x, y) -> this.info = null, HandlerPriority.HIGH); + } + + public ResourceLocation name() { + return this.getInfo().getName(); + } + + public ClientLevel level() { + return GameUtils.getWorld().orElseThrow(); + } + + public int seaLevel() { + return this.getInfo().getSeaLevel(); + } + + public boolean alwaysOutside() { + return this.getInfo().alwaysOutside(); + } + + public int getSpaceHeight() { + return this.getInfo().getSpaceHeight(); + } + + public int getCloudHeight() { + return this.getInfo().getCloudHeight(); + } + + public boolean getCompassWobble() { + return this.getInfo().getCompassWobble(); + } + + private DimensionInfo getInfo() { + if (this.info == null) + this.info = this.dimensionLibrary.getData(this.level()); + return this.info; + } +} diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionLibrary.java index 05a590d6..7a9959f4 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/DimensionLibrary.java @@ -5,7 +5,7 @@ import net.minecraft.resources.ResourceKey; import net.minecraft.world.level.Level; import org.orecruncher.dsurround.config.data.DimensionConfigRule; -import org.orecruncher.dsurround.config.dimension.DimensionInfo; +import org.orecruncher.dsurround.config.DimensionInfo; import org.orecruncher.dsurround.config.libraries.IDimensionLibrary; import org.orecruncher.dsurround.config.libraries.IReloadEvent; import org.orecruncher.dsurround.lib.collections.ObjectArray; diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/ItemLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/ItemLibrary.java index ca12910e..9a887fa8 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/ItemLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/ItemLibrary.java @@ -10,6 +10,7 @@ import net.minecraft.world.item.Item; import net.minecraft.world.item.*; import org.jetbrains.annotations.Nullable; +import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.config.ItemClassType; import org.orecruncher.dsurround.config.libraries.IItemLibrary; import org.orecruncher.dsurround.config.libraries.IReloadEvent; @@ -32,14 +33,16 @@ public class ItemLibrary implements IItemLibrary { private final ITagLibrary tagLibrary; private final IModLog logger; + private final Configuration config; private final Reference2ObjectOpenHashMap itemEquipFactories = new Reference2ObjectOpenHashMap<>(); private final Reference2ObjectOpenHashMap itemSwingFactories = new Reference2ObjectOpenHashMap<>(); private final Reference2ObjectOpenHashMap itemArmorStepFactories = new Reference2ObjectOpenHashMap<>(); private int version; - public ItemLibrary(ITagLibrary tagLibrary, IModLog logger) { + public ItemLibrary(ITagLibrary tagLibrary, Configuration config, IModLog logger) { this.tagLibrary = tagLibrary; this.logger = ModLog.createChild(logger, "ItemLibrary"); + this.config = config; } @Override @@ -98,7 +101,7 @@ public Stream dump() { if (itemEquipSound != null) return SoundFactoryBuilder .create(itemEquipSound) - .category(SoundSource.PLAYERS).volume(0.5F).pitch(0.8F, 1.2F).build(); + .category(SoundSource.PLAYERS).volume(0.25F).pitch(0.8F, 1.2F).build(); return defaultSoundFactory.get(); } @@ -107,29 +110,33 @@ public Stream dump() { @Nullable private static SoundEvent getEquipableSoundEvent(ItemStack stack) { - var item = stack.getItem(); SoundEvent itemEquipSound = null; - - if (item instanceof Equipable equipment) - itemEquipSound = equipment.getEquipSound().value(); - else if (item instanceof ArmorItem armor) - itemEquipSound = armor.getEquipSound().value(); - + var equipable = Equipable.get(stack); + if (equipable != null) { + itemEquipSound = equipable.getEquipSound().value(); + } return itemEquipSound; } @Nullable private SoundEvent getSoundEvent(ItemStack stack) { - // Look for special Equipment and ArmorItem types since they may have built in equip sounds + // Look for special Equipment and ArmorItem types since they may have built in equipped sounds SoundEvent itemEquipSound = getEquipableSoundEvent(stack); if (itemEquipSound != null) return itemEquipSound; - var item = stack.getItem(); + if (this.config.entityEffects.enableToolbarBlockSounds) { + Item item = stack.getItem(); + if (item instanceof BlockItem blockItem) { + var soundType = blockItem.getBlock().defaultBlockState().getSoundType(); + itemEquipSound = soundType.getStepSound(); + } + } + + if (itemEquipSound != null) + return itemEquipSound; - if (item instanceof ElytraItem elytraItem) - itemEquipSound = elytraItem.getEquipSound().value(); - else if (this.tagLibrary.is(ItemTags.LAVA_BUCKETS, stack)) + if (this.tagLibrary.is(ItemTags.LAVA_BUCKETS, stack)) itemEquipSound = SoundEvents.BUCKET_FILL_LAVA; else if (this.tagLibrary.is(ItemTags.WATER_BUCKETS, stack)) itemEquipSound = SoundEvents.BUCKET_FILL; diff --git a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/SoundLibrary.java b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/SoundLibrary.java index 88567edb..7a0affd5 100644 --- a/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/SoundLibrary.java +++ b/common/src/main/java/org/orecruncher/dsurround/config/libraries/impl/SoundLibrary.java @@ -1,22 +1,33 @@ package org.orecruncher.dsurround.config.libraries.impl; +import com.google.common.collect.ImmutableSet; import com.mojang.serialization.Codec; import com.mojang.serialization.codecs.UnboundedMapCodec; import it.unimi.dsi.fastutil.objects.Object2ObjectOpenHashMap; import it.unimi.dsi.fastutil.objects.ObjectOpenHashSet; +import net.minecraft.client.resources.sounds.SoundInstance; +import net.minecraft.core.BlockPos; +import net.minecraft.core.Direction; import net.minecraft.core.registries.BuiltInRegistries; import net.minecraft.resources.ResourceLocation; import net.minecraft.sounds.Music; import net.minecraft.sounds.SoundEvent; +import net.minecraft.sounds.SoundEvents; import net.minecraft.sounds.SoundSource; +import net.minecraft.world.level.block.state.BlockState; +import org.jetbrains.annotations.Nullable; import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.Constants; import org.orecruncher.dsurround.config.IndividualSoundConfigEntry; +import org.orecruncher.dsurround.config.SoundMapping; +import org.orecruncher.dsurround.config.data.SoundMappingConfigRule; import org.orecruncher.dsurround.config.data.SoundMetadataConfig; import org.orecruncher.dsurround.config.libraries.IReloadEvent; import org.orecruncher.dsurround.config.libraries.ISoundLibrary; +import org.orecruncher.dsurround.gui.sound.ConfigSoundInstance; import org.orecruncher.dsurround.lib.CodecExtensions; import org.orecruncher.dsurround.lib.Comparers; +import org.orecruncher.dsurround.lib.GameUtils; import org.orecruncher.dsurround.lib.Library; import org.orecruncher.dsurround.lib.logging.IModLog; import org.orecruncher.dsurround.lib.logging.ModLog; @@ -45,13 +56,19 @@ public final class SoundLibrary implements ISoundLibrary { private static final String SOUNDS_JSON = "sounds.json"; private static final String FACTORY_JSON = "sound_factories.json"; private static final String SOUND_CONFIG_FILE = "soundconfig.json"; + private static final String SOUND_MAPPING_JSON = "sound_mappings.json"; private static final UnboundedMapCodec SOUND_FILE_CODEC = Codec.unboundedMap(Codec.STRING, SoundMetadataConfig.CODEC); private static final Codec> FACTORY_FILE_CODEC = Codec.list(SoundFactory.CODEC); private static final Codec> SOUND_CONFIG_CODEC = Codec.list(IndividualSoundConfigEntry.CODEC); + private static final Codec> SOUND_MAPPING_CODEC = Codec.list(SoundMappingConfigRule.CODEC); private static final ResourceLocation MISSING_RESOURCE = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "missing_sound"); private static final SoundEvent MISSING = SoundEvent.createVariableRangeEvent(MISSING_RESOURCE); + private static final ResourceLocation THUNDER_SOUND = SoundEvents.LIGHTNING_BOLT_THUNDER.getLocation(); + private static final Set SOUND_REMAP_BLOCKED_MOBS = ImmutableSet.of("creeper"); + private static final BlockPos.MutableBlockPos MUTABLE_BLOCK_POS = new BlockPos.MutableBlockPos(); + private final IModLog logger; private final Configuration config; private final Path soundConfigPath; @@ -63,6 +80,7 @@ public final class SoundLibrary implements ISoundLibrary { private final Set blockedSounds = new ObjectOpenHashSet<>(); private final Set culledSounds = new ObjectOpenHashSet<>(); private final List startupSounds = new ArrayList<>(); + private final Map soundRemappings = new Object2ObjectOpenHashMap<>(); private List soundConfiguration = new ArrayList<>(); public SoundLibrary(Configuration config, IModLog logger, IMinecraftDirectories directories) { @@ -91,6 +109,7 @@ public void reload(ResourceUtilities resourceUtilities, IReloadEvent.Scope scope this.myRegistry.clear(); this.soundMetadata.clear(); this.soundFactories.clear(); + this.soundRemappings.clear(); this.loadSoundConfiguration(); // Initializes the internal sound registry once all the other mods have @@ -107,8 +126,13 @@ public void reload(ResourceUtilities resourceUtilities, IReloadEvent.Scope scope var findResults = resourceUtilities.findModResources(FACTORY_FILE_CODEC, FACTORY_JSON); findResults.forEach(this::registerSoundFactories); + + var soundMappings = resourceUtilities.findModResources(SOUND_MAPPING_CODEC, SOUND_MAPPING_JSON); + soundMappings.forEach(this::registerSoundRemappings); + this.logger.info("Number of SoundEvents cached: %d", this.myRegistry.size()); this.logger.info("Number of factories cached: %d", this.soundFactories.size()); + this.logger.info("Number of sound remappings cached: %d", this.soundRemappings.size()); } @Override @@ -209,6 +233,84 @@ public void saveIndividualSoundConfigs(Collection co this.save(); } + @Override + public Optional remapSound(SoundInstance soundInstance) { + + // Sounds played from the sound config menu are not remapped + if (soundInstance instanceof ConfigSoundInstance) + return Optional.empty(); + + var soundLocation = soundInstance.getLocation(); + + // If it is a thunder sound, and replacement is not enabled, don't do anything + if (THUNDER_SOUND.equals(soundLocation)) { + if (!this.config.soundOptions.replaceThunderSounds) + return Optional.empty(); + } else if (!this.config.soundOptions.remapSounds) { + // If it is not a thunder sound and remap is disabled + return Optional.empty(); + } + + var mappingRule = this.soundRemappings.get(soundLocation); + + if (mappingRule != null) { + // Get the BlockState of the block below the sound location if needed + @Nullable + BlockState blockState = null; + if (mappingRule.isBlockStateNeeded()) { + var level = GameUtils.getWorld().orElseThrow(); + var pos = BlockPos.containing(soundInstance.getX(), soundInstance.getY() + 0.25D, soundInstance.getZ()).below(); + blockState = level.getBlockState(pos); + + // If the BlockState is air or not solid, it means we are hanging at a block edge. Scan around looking for + // something that is solid. It's not perfect, but it's 90% down the center. + if (blockState.isAir() || !blockState.isSolid()) { + for (var dir : Direction.Plane.HORIZONTAL) { + blockState = level.getBlockState(MUTABLE_BLOCK_POS.setWithOffset(pos, dir)); + if (!blockState.isAir() && blockState.isSolid()) + break; + } + } + } + + var soundFactory = mappingRule + .findMatch(blockState) + .map(this::getSoundFactoryOrDefault); + + if (soundFactory.isPresent()) { + // Need to force a sound to be selected so we can get the volume + soundInstance.resolve(GameUtils.getSoundManager()); + var volumeScale = soundInstance.getVolume(); + return Optional.of(soundFactory.get().createAtLocation(soundInstance.getX(), soundInstance.getY(), soundInstance.getZ(), volumeScale)); + } + } + + return Optional.empty(); + } + + /** + * Examines the sound location information to determine if it is a mob step sound, and then remaps to a block + * sound similar to what happens with the player. + */ + @Nullable + private ResourceLocation remapMobStepSound(SoundInstance soundInstance) { + var soundLocation = soundInstance.getLocation(); + var path = soundLocation.getPath(); + if (path.startsWith("entity.") && path.endsWith("step")) { + // Get the mob this sound is for. We do not want to convert mobs like creepers. + var mobType = path.substring(7, path.indexOf('.', 7)); + if (!SOUND_REMAP_BLOCKED_MOBS.contains(mobType)) { + var level = GameUtils.getWorld().orElseThrow(); + var pos = BlockPos.containing(soundInstance.getX(), soundInstance.getY(), soundInstance.getZ()).below(); + soundLocation = level.getBlockState(pos).getSoundType().getStepSound().getLocation(); + this.logger.debug("Mob sound remapping from %s to %s", soundInstance.getLocation(), soundLocation); + return soundLocation; + } + } + + return null; + } + private void registerSoundFile(DiscoveredResource> soundFile) { var result = soundFile.resourceContent(); result.forEach((key, value) -> { @@ -225,6 +327,19 @@ private void registerSoundFile(DiscoveredResource> mappings) { + for (var mapping : mappings.resourceContent()) { + if (!this.soundRemappings.containsKey(mapping.soundEvent())) { + this.soundRemappings.put(mapping.soundEvent(), SoundMapping.of(mapping)); + } else { + // Need to merge rules + var existingMapping = this.soundRemappings.get(mapping.soundEvent()); + existingMapping.merge(mapping); + } + } + this.logger.debug("Registered %d sound remappings for namespace %s", mappings.resourceContent().size(), mappings.namespace()); + } + private void registerSoundFactories(DiscoveredResource> factories) { factories.resourceContent().forEach(factory -> this.soundFactories.put(factory.getLocation(), factory)); this.logger.debug("Registered %d sound factories for namespace %s", factories.resourceContent().size(), factories.namespace()); diff --git a/common/src/main/java/org/orecruncher/dsurround/effects/WaterRippleHandler.java b/common/src/main/java/org/orecruncher/dsurround/effects/WaterRippleHandler.java index 46ad4cca..1c6b15ed 100644 --- a/common/src/main/java/org/orecruncher/dsurround/effects/WaterRippleHandler.java +++ b/common/src/main/java/org/orecruncher/dsurround/effects/WaterRippleHandler.java @@ -7,8 +7,8 @@ import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.config.WaterRippleStyle; import org.orecruncher.dsurround.config.libraries.ITagLibrary; +import org.orecruncher.dsurround.effects.particles.ParticleRenderCollection; import org.orecruncher.dsurround.effects.particles.WaterRippleParticle; -import org.orecruncher.dsurround.lib.GameUtils; import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.tags.FluidTags; @@ -17,6 +17,8 @@ public class WaterRippleHandler { private static final Configuration.BlockEffects CONFIG = ContainerManager.resolve(Configuration.BlockEffects.class); private static final ITagLibrary TAG_LIBRARY = ContainerManager.resolve(ITagLibrary.class); + private static final ParticleRenderCollection.Helper rippleHelper = + new ParticleRenderCollection.Helper<>("WaterRipples", () -> CONFIG.waterRippleStyle.getTexture()); // Fudge factor because the height algo is off. private static final double LIQUID_HEIGHT_ADJUST = (1D / 9D) + 0.1D; @@ -26,10 +28,8 @@ private static boolean doRipples() { } private static void addWaterRipple(ClientLevel world, double x, double y, double z) { - var ripple = new WaterRippleParticle( - CONFIG.waterRippleStyle, - world, x, y, z); - GameUtils.getParticleManager().add(ripple); + var ripple = new WaterRippleParticle(CONFIG.waterRippleStyle, world, x, y, z); + rippleHelper.add(ripple); } public static void createRippleParticle(ClientLevel world, Particle particle, Vec3 position) { diff --git a/common/src/main/java/org/orecruncher/dsurround/effects/entity/BreathEffect.java b/common/src/main/java/org/orecruncher/dsurround/effects/entity/BreathEffect.java index 2aea6e9c..b6430e65 100644 --- a/common/src/main/java/org/orecruncher/dsurround/effects/entity/BreathEffect.java +++ b/common/src/main/java/org/orecruncher/dsurround/effects/entity/BreathEffect.java @@ -2,7 +2,6 @@ import net.minecraft.core.BlockPos; import net.minecraft.world.entity.LivingEntity; -import net.minecraft.world.level.Level; import net.minecraft.world.level.block.state.BlockState; import org.orecruncher.dsurround.effects.particles.FrostBreathParticle; import org.orecruncher.dsurround.lib.GameUtils; @@ -10,7 +9,6 @@ import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; import org.orecruncher.dsurround.lib.system.ITickCount; import org.orecruncher.dsurround.lib.random.MurmurHash3; -import org.orecruncher.dsurround.lib.world.WorldUtils; public class BreathEffect extends EntityEffectBase { @@ -83,8 +81,7 @@ protected boolean showWaterBubbles(final BlockState headBlock) { protected boolean showFrostBreath(final LivingEntity entity, final BlockState headBlock, final BlockPos pos) { if (headBlock.isAir()) { - final Level world = entity.level(); - return SEASONAL_INFORMATION.isColdTemperature(world, pos); + return SEASONAL_INFORMATION.isColdTemperature(pos); } return false; } diff --git a/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleRenderCollection.java b/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleRenderCollection.java new file mode 100644 index 00000000..05dbfdc8 --- /dev/null +++ b/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleRenderCollection.java @@ -0,0 +1,173 @@ +package org.orecruncher.dsurround.effects.particles; + +import com.mojang.blaze3d.systems.RenderSystem; +import com.mojang.blaze3d.vertex.VertexConsumer; +import dev.architectury.event.events.client.ClientLifecycleEvent; +import net.minecraft.client.Camera; +import net.minecraft.client.Minecraft; +import net.minecraft.client.multiplayer.ClientLevel; +import net.minecraft.client.particle.Particle; +import net.minecraft.client.particle.ParticleRenderType; +import net.minecraft.client.particle.TextureSheetParticle; +import net.minecraft.resources.ResourceLocation; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; +import org.orecruncher.dsurround.eventing.ClientEventHooks; +import org.orecruncher.dsurround.eventing.ClientState; +import org.orecruncher.dsurround.eventing.CollectDiagnosticsEvent; +import org.orecruncher.dsurround.lib.GameUtils; +import org.orecruncher.dsurround.lib.collections.ObjectArray; + +import java.lang.ref.WeakReference; +import java.util.Objects; +import java.util.function.Consumer; +import java.util.function.Supplier; + +/** + * Special particle that proxies a collection in the particle engine. The commonality + * of the collection is rendering setup. This collection is centered on the player + * to prevent it from going out of scope. It is modeled on the NoRenderParticle in + * Minecraft. + */ +public final class ParticleRenderCollection extends Particle { + + private final Consumer setup; + private final Supplier textureSupplier; + private final ObjectArray particles; + + private ParticleRenderCollection(@NotNull ClientLevel clientLevel, @NotNull Supplier textureSupplier, @Nullable Consumer setup) { + super(clientLevel, 0, 0, 0); + this.setup = Objects.requireNonNullElseGet(setup, () -> this::standardSetup); + this.textureSupplier = textureSupplier; + this.particles = new ObjectArray<>(128); + this.tick(); + } + + @NotNull + @Override + public ParticleRenderType getRenderType() { + // Can't use NO_RENDER as the ParticleEngine will not attempt to render + return ParticleRenderType.CUSTOM; + } + + @Override + public void tick() { + // Keep the particle colocated with the player + var playerPos = GameUtils.getPlayer().orElseThrow().getEyePosition(); + this.setPos(playerPos.x(), playerPos.y(), playerPos.z()); + + if (!this.particles.isEmpty()) { + this.particles.forEach(Particle::tick); + this.particles.removeIf(p -> !p.isAlive()); + } + } + + @Override + public void render(@NotNull VertexConsumer vertexConsumer, @NotNull Camera camera, float tickDelta) { + if (this.particles.isEmpty()) + return; + + RenderSystem.setShaderTexture(0, this.textureSupplier.get()); + this.setup.accept(camera); + this.particles.forEach(p -> p.render(vertexConsumer, camera, tickDelta)); + } + + private void standardSetup(@NotNull Camera camera) { + RenderSystem.depthMask(true); + RenderSystem.enableBlend(); + RenderSystem.defaultBlendFunc(); + } + + public void add(@NotNull TParticle particle) { + // Can only accept custom style particles + if (particle.getRenderType() != this.getRenderType()) + throw new RuntimeException("Can only add render type %s particles to collection".formatted(this.getRenderType())); + this.particles.add(particle); + } + + /** + * Helper that manages related particles in Minecraft's ParticleEngine. The helper will register with events, so + * instances of this class need to be maintained as singletons throughout the lifetime of the client. + */ + public static final class Helper { + + private final String name; + private final Consumer setup; + private final Supplier textureSupplier; + + private WeakReference> particle; + private String diagnostics; + + /** + * Initializes a helper instance used to manage the state of the main particle within the ParticleEngine. + * Particle rendering will use the default setup. + * + * @param name The name of the helper; used in diagnostics + * @param textureSupplier Provides the texture to bind when rendering + */ + public Helper(@NotNull String name, @NotNull Supplier textureSupplier) { + this(name, textureSupplier, null); + } + + /** + * Initializes a helper instance used to manage the state of the main particle within the ParticleEngine. + * + * @param name The name of the helper; used in diagnostics + * @param textureSupplier Provides the texture to bind when rendering + * @param setup Provides for the configuration of the rendering system if the default is not enough + */ + public Helper(@NotNull String name, @NotNull Supplier textureSupplier, @Nullable Consumer setup) { + this.name = name; + this.setup = setup; + this.textureSupplier = textureSupplier; + this.diagnostics = this.name; + + ClientLifecycleEvent.CLIENT_LEVEL_LOAD.register(state -> this.clear()); + ClientEventHooks.COLLECT_DIAGNOSTICS.register(this::collectDiagnostics); + ClientState.TICK_END.register(this::tick); + } + + /** + * Adds a particle to the helper. + * + * @param particle The particle to add + */ + public void add(TParticle particle) { + this.get().add(particle); + } + + @NotNull + private ParticleRenderCollection get() { + var pc = this.particle != null ? this.particle.get() : null; + if (pc == null || !pc.isAlive()) { + pc = new ParticleRenderCollection<>(GameUtils.getWorld().orElseThrow(), this.textureSupplier, this.setup); + this.particle = new WeakReference<>(pc); + GameUtils.getParticleManager().add(pc); + } + return pc; + } + + private void clear() { + var pc = this.particle != null ? this.particle.get() : null; + if (pc != null) { + pc.remove(); + this.particle = null; + } + } + + private void tick(@NotNull Minecraft client) { + var pc = this.particle != null ? this.particle.get() : null; + this.diagnostics = this.name + ": "; + if (pc == null) + this.diagnostics += "Not Set"; + else if (!pc.isAlive()) + this.diagnostics += "DEAD"; + else + this.diagnostics += pc.particles.size(); + } + + private void collectDiagnostics(@NotNull CollectDiagnosticsEvent event) { + event.add(CollectDiagnosticsEvent.Section.Particles, this.diagnostics); + } + } +} diff --git a/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleSheets.java b/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleSheets.java deleted file mode 100644 index 5062006a..00000000 --- a/common/src/main/java/org/orecruncher/dsurround/effects/particles/ParticleSheets.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.orecruncher.dsurround.effects.particles; - -import net.minecraft.client.renderer.texture.SimpleTexture; -import net.minecraft.resources.ResourceLocation; -import org.orecruncher.dsurround.Constants; -import org.orecruncher.dsurround.Configuration; -import org.orecruncher.dsurround.lib.GameUtils; -import org.orecruncher.dsurround.lib.di.ContainerManager; -import org.orecruncher.dsurround.mixins.core.MixinParticleManager; - -import java.util.ArrayList; - -public final class ParticleSheets { - - private static Configuration.BlockEffects CONFIG; - - public static final ResourceLocation TEXTURE_WATER_RIPPLE_PIXELATED_CIRCLE = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "textures/particles/pixel_ripples.png"); - - public static final net.minecraft.client.particle.ParticleRenderType RIPPLE_RENDER = - new DsurroundParticleRenderType(TEXTURE_WATER_RIPPLE_PIXELATED_CIRCLE) { - @Override - protected ResourceLocation getTexture() { - return CONFIG.waterRippleStyle.getTexture(); - } - }; - - public static void register() { - - CONFIG = ContainerManager.resolve(Configuration.BlockEffects.class); - var manager = GameUtils.getTextureManager(); - manager.register(TEXTURE_WATER_RIPPLE_PIXELATED_CIRCLE, new SimpleTexture(TEXTURE_WATER_RIPPLE_PIXELATED_CIRCLE)); - - var existingSheets = MixinParticleManager.dsurround_getParticleTextureSheets(); - assert existingSheets != null; - existingSheets = new ArrayList<>(existingSheets); - existingSheets.add(RIPPLE_RENDER); - MixinParticleManager.dsurround_setParticleTextureSheets(existingSheets); - } -} diff --git a/common/src/main/java/org/orecruncher/dsurround/effects/particles/WaterRippleParticle.java b/common/src/main/java/org/orecruncher/dsurround/effects/particles/WaterRippleParticle.java index 9402d84e..7c03774e 100644 --- a/common/src/main/java/org/orecruncher/dsurround/effects/particles/WaterRippleParticle.java +++ b/common/src/main/java/org/orecruncher/dsurround/effects/particles/WaterRippleParticle.java @@ -71,7 +71,7 @@ public WaterRippleParticle(WaterRippleStyle rippleStyle, ClientLevel world, doub @Override public @NotNull ParticleRenderType getRenderType() { - return ParticleSheets.RIPPLE_RENDER; + return ParticleRenderType.CUSTOM; } @Override @@ -101,6 +101,7 @@ protected float getV1() { @Override public void render(VertexConsumer vertexConsumer, Camera camera, float tickDelta) { + Vec3 vec3d = camera.getPosition(); float X = (float)(Mth.lerp(tickDelta, this.xo, this.x) - vec3d.x()); float Y = (float)(Mth.lerp(tickDelta, this.yo, this.y) - vec3d.y()); diff --git a/common/src/main/java/org/orecruncher/dsurround/eventing/CollectDiagnosticsEvent.java b/common/src/main/java/org/orecruncher/dsurround/eventing/CollectDiagnosticsEvent.java index 7bc816e0..4d55e0cd 100644 --- a/common/src/main/java/org/orecruncher/dsurround/eventing/CollectDiagnosticsEvent.java +++ b/common/src/main/java/org/orecruncher/dsurround/eventing/CollectDiagnosticsEvent.java @@ -12,6 +12,7 @@ public final class CollectDiagnosticsEvent { public enum Section { Header(false), Systems, + Particles, Timers(false), Environment(false), Emitters, diff --git a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/ClockOverlay.java b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/ClockOverlay.java index a634748f..a13495ef 100644 --- a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/ClockOverlay.java +++ b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/ClockOverlay.java @@ -58,7 +58,7 @@ public void tick(Minecraft client) { this.clockDisplay.clear(); this.clockDisplay.add(this.clock.getFormattedTime()); - this.seasonalInformation.getCurrentSeasonTranslated(player.level()).ifPresent(this.clockDisplay::add); + this.seasonalInformation.getCurrentSeasonTranslated().ifPresent(this.clockDisplay::add); var textRender = GameUtils.getTextRenderer(); this.renderWidth = textRender.width(this.clockDisplay.get(0)); diff --git a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/CompassOverlay.java b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/CompassOverlay.java index 29ed7556..c9a2f63b 100644 --- a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/CompassOverlay.java +++ b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/CompassOverlay.java @@ -11,8 +11,10 @@ import org.joml.Matrix4f; import org.orecruncher.dsurround.Constants; import org.orecruncher.dsurround.Configuration; +import org.orecruncher.dsurround.config.libraries.IDimensionInformation; import org.orecruncher.dsurround.config.libraries.ITagLibrary; import org.orecruncher.dsurround.lib.GameUtils; +import org.orecruncher.dsurround.lib.random.Randomizer; import org.orecruncher.dsurround.tags.ItemEffectTags; public class CompassOverlay extends AbstractOverlay { @@ -30,14 +32,19 @@ public class CompassOverlay extends AbstractOverlay { private static final ResourceLocation COMPASS_TEXTURE = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "textures/compass.png"); private final ITagLibrary tagLibrary; + private final IDimensionInformation dimensionInformation; private final Configuration config; + private final CompassWobble wobbler; private boolean showCompass; + private boolean spinRandomly; private float scale; private float spriteOffset; - public CompassOverlay(Configuration config, ITagLibrary tagLibrary) { + public CompassOverlay(Configuration config, ITagLibrary tagLibrary, IDimensionInformation dimensionInformation) { this.tagLibrary = tagLibrary; + this.dimensionInformation = dimensionInformation; this.config = config; + this.wobbler = new CompassWobble(); this.showCompass = false; this.spriteOffset = this.config.compassAndClockOptions.compassStyle.getSpriteNumber(); this.scale = (float)this.config.compassAndClockOptions.scale; @@ -53,7 +60,20 @@ public void tick(Minecraft client) { var player = GameUtils.getPlayer().orElseThrow(); var mainHandItem = player.getMainHandItem(); var offHandItem = player.getOffhandItem(); - this.showCompass = this.doShowCompass(mainHandItem) || this.doShowCompass(offHandItem); + + var mainHandShow = this.doShowCompass(mainHandItem); + var offHandShow = this.doShowCompass(offHandItem); + this.showCompass = mainHandShow || offHandShow; + + if (mainHandShow) { + this.spinRandomly = this.doCompassSpin(mainHandItem); + } + + if (offHandShow && !this.spinRandomly) { + this.spinRandomly = this.doCompassSpin(offHandItem); + } + + this.wobbler.update(player.level().getGameTime()); } } @@ -61,6 +81,10 @@ private boolean doShowCompass(ItemStack stack) { return this.tagLibrary.is(ItemEffectTags.COMPASSES, stack); } + private boolean doCompassSpin(ItemStack stack) { + return this.dimensionInformation.getCompassWobble() && this.tagLibrary.is(ItemEffectTags.COMPASS_WOBBLE, stack); + } + @Override public void render(GuiGraphics context, float partialTick) { if (!this.showCompass) @@ -72,9 +96,16 @@ public void render(GuiGraphics context, float partialTick) { matrixStack.pushPose(); - final var player = GameUtils.getPlayer().orElseThrow(); + float rotation; - int direction = Mth.floor(((player.getViewYRot(partialTick) * TEXTURE_SIZE) / 360F) + 0.5D) & (TEXTURE_SIZE - 1); + if (this.spinRandomly) { + rotation = this.wobbler.getRandomlySpinningRotation(partialTick); + } else { + final var player = GameUtils.getPlayer().orElseThrow(); + rotation = player.getViewYRot(partialTick); + } + + int direction = Mth.floor(((rotation * TEXTURE_SIZE) / 360F) + 0.5D) & (TEXTURE_SIZE - 1); float x = (context.guiWidth() - BAND_WIDTH * this.scale) / 2F; float y = (context.guiHeight() - CROSSHAIR_OFFSET - BAND_HEIGHT * this.scale) / 2F; @@ -117,4 +148,56 @@ void drawTexturedQuad(PoseStack stack, ResourceLocation texture, float x1, float if (mesh != null) BufferUploader.drawWithShader(mesh); } + + /** + * Cloned from the Minecraft compass code + */ + static class CompassWobble { + private static int TICK_DELAY = 5; + private static float MAX_DELTA_TICK = 1F / 20F; + private float targetRotation; + private float lastRotation; + private float rotation; + private float rotationStep; + private long lastUpdateTick; + private int tickWait; + + public float getRandomlySpinningRotation(float partialTick) { + return Mth.lerp(partialTick, this.lastRotation, this.rotation) * 360F; + } + + public void update(long tickCount) { + if (this.lastUpdateTick != tickCount) { + this.updateRotation(tickCount); + } + } + + private void updateRotation(long tickCount) { + this.lastUpdateTick = tickCount; + this.lastRotation = this.rotation; + + if (this.rotation < this.targetRotation) { + this.rotation += this.rotationStep; + if (this.rotation > this.targetRotation) { + this.rotation = this.targetRotation; + } + } else if (this.rotation > this.targetRotation) { + this.rotation -= this.rotationStep; + if (this.rotation < this.targetRotation) { + this.rotation = this.targetRotation; + } + } + + // If we reached the target rotation, we need to set a new one + // and calculate the stepping to get there. + if (this.rotation == this.targetRotation) { + if (this.tickWait < 0 ) { + this.tickWait = Randomizer.current().nextInt(TICK_DELAY); + } else if (--this.tickWait == 0) { + this.targetRotation = Randomizer.current().nextFloat(); + this.rotationStep = MAX_DELTA_TICK; + } + } + } + } } diff --git a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/DiagnosticsOverlay.java b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/DiagnosticsOverlay.java index b8c3c741..32d71fa6 100644 --- a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/DiagnosticsOverlay.java +++ b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/DiagnosticsOverlay.java @@ -37,6 +37,7 @@ public class DiagnosticsOverlay extends AbstractOverlay { static { COLOR_MAP.put(CollectDiagnosticsEvent.Section.Header, ColorPalette.PUMPKIN_ORANGE); COLOR_MAP.put(CollectDiagnosticsEvent.Section.Systems, ColorPalette.GREEN); + COLOR_MAP.put(CollectDiagnosticsEvent.Section.Particles, ColorPalette.HOT_PINK); COLOR_MAP.put(CollectDiagnosticsEvent.Section.Timers, ColorPalette.KEY_LIME); COLOR_MAP.put(CollectDiagnosticsEvent.Section.Environment, ColorPalette.AQUAMARINE); COLOR_MAP.put(CollectDiagnosticsEvent.Section.Emitters, ColorPalette.SEASHELL); @@ -51,6 +52,7 @@ public class DiagnosticsOverlay extends AbstractOverlay { LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Header); LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Environment); LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Systems); + LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Particles); LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Emitters); LEFT_SIDE_LAYOUT.add(CollectDiagnosticsEvent.Section.Sounds); diff --git a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/OverlayManager.java b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/OverlayManager.java index 2b2b032f..9c1a2c28 100644 --- a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/OverlayManager.java +++ b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/OverlayManager.java @@ -1,5 +1,6 @@ package org.orecruncher.dsurround.gui.overlay; +import net.minecraft.client.DeltaTracker; import net.minecraft.client.Minecraft; import net.minecraft.client.gui.GuiGraphics; import org.orecruncher.dsurround.lib.collections.ObjectArray; @@ -21,7 +22,8 @@ public OverlayManager() { ClientState.TICK_END.register(this::tick); } - public void render(GuiGraphics context, float partialTick) { + public void render(GuiGraphics context, DeltaTracker deltaTracker) { + var partialTick = deltaTracker.getGameTimeDeltaTicks(); this.overlays.forEach(overlay -> overlay.render(context, partialTick)); } diff --git a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/plugins/RuntimeDiagnosticsPlugin.java b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/plugins/RuntimeDiagnosticsPlugin.java index f455b1c7..d2e26b36 100644 --- a/common/src/main/java/org/orecruncher/dsurround/gui/overlay/plugins/RuntimeDiagnosticsPlugin.java +++ b/common/src/main/java/org/orecruncher/dsurround/gui/overlay/plugins/RuntimeDiagnosticsPlugin.java @@ -43,7 +43,7 @@ public void onCollect(CollectDiagnosticsEvent event) { this.clock.update(world); event.add(CollectDiagnosticsEvent.Section.Header, this.clock.getFormattedTime()); - var seasonInfo = this.seasonalInformation.getCurrentSeasonTranslated(world).orElse(Component.literal("UNKNOWN")); + var seasonInfo = this.seasonalInformation.getCurrentSeasonTranslated().orElse(Component.literal("UNKNOWN")); var seasonText = Component.translatable("Season: %s (%s)", seasonInfo, this.seasonalInformation.getProviderName()); event.add(CollectDiagnosticsEvent.Section.Header, seasonText); diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/BlockPosUtil.java b/common/src/main/java/org/orecruncher/dsurround/lib/BlockPosUtil.java deleted file mode 100644 index 3c862368..00000000 --- a/common/src/main/java/org/orecruncher/dsurround/lib/BlockPosUtil.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.orecruncher.dsurround.lib; - -import net.minecraft.core.BlockPos; - -public final class BlockPosUtil { - - public static boolean canFormCuboid(final BlockPos p1, final BlockPos p2) { - return !(p1.getX() == p2.getX() && p1.getZ() == p2.getZ() && p1.getY() == p2.getY()); - } - - public static BlockPos createMinPoint( final BlockPos p1, final BlockPos p2) { - return new BlockPos(Math.min(p1.getX(), p2.getX()), Math.min(p1.getY(), p2.getY()), - Math.min(p1.getZ(), p2.getZ())); - } - - public static BlockPos createMaxPoint( final BlockPos p1, final BlockPos p2) { - return new BlockPos(Math.max(p1.getX(), p2.getX()), Math.max(p1.getY(), p2.getY()), - Math.max(p1.getZ(), p2.getZ())); - } - - /** - * Determines if the test point is contained within the volume described by two - * other points. It is expected that the calling routine has ensured that the - * min/max points are valid. If they are not valid, the results will more than - * likely be erroneous. - * - * @param test The point that is being tested - * @param min The point describing the minimum vertex of the volume - * @param max The point describing the maximum vertex of the volume - * @return Whether the test point is within the boundaries of the volume, - * inclusive - */ - public static boolean contains( final BlockPos test, final BlockPos min, - final BlockPos max) { - return test.getX() >= min.getX() && test.getX() <= max.getX() - && test.getY() >= min.getY() && test.getY() <= max.getY() - && test.getZ() >= min.getZ() && test.getZ() <= max.getZ(); - } -} \ No newline at end of file diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/Library.java b/common/src/main/java/org/orecruncher/dsurround/lib/Library.java index 2ac12af1..079546d7 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/Library.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/Library.java @@ -7,8 +7,6 @@ import org.orecruncher.dsurround.lib.platform.*; import org.orecruncher.dsurround.eventing.ClientState; import org.orecruncher.dsurround.lib.logging.IModLog; -import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; -import org.orecruncher.dsurround.lib.seasons.SeasonManager; import org.orecruncher.dsurround.lib.system.ISystemClock; import org.orecruncher.dsurround.lib.system.ITickCount; import org.orecruncher.dsurround.lib.system.SystemClock; @@ -52,7 +50,6 @@ private static void configureServiceDependencies() { .registerSingleton(ISystemClock.class, SystemClock.class) .registerSingleton(IMinecraftDirectories.class, modInfo) .registerSingleton(IClientTasking.class, ClientTasking.class) - .registerSingleton(ITickCount.class, TickCounter.class) - .registerSingleton(ISeasonalInformation.class, SeasonManager.HANDLER); + .registerSingleton(ITickCount.class, TickCounter.class); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/WeightTable.java b/common/src/main/java/org/orecruncher/dsurround/lib/WeightTable.java index c0eeb7b4..05e9a122 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/WeightTable.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/WeightTable.java @@ -1,10 +1,10 @@ package org.orecruncher.dsurround.lib; -import org.orecruncher.dsurround.lib.collections.ObjectArray; +import net.minecraft.util.random.WeightedEntry; +import net.minecraft.util.random.WeightedRandom; import org.orecruncher.dsurround.lib.random.IRandomizer; import org.orecruncher.dsurround.lib.random.Randomizer; -import java.util.Collection; import java.util.List; import java.util.Optional; import java.util.stream.Stream; @@ -12,58 +12,27 @@ /** * Classic WeightTable for random weighted selection. */ -@SuppressWarnings("unused") public class WeightTable { public static Optional makeSelection(final Stream> inputStream) { - return makeSelection(inputStream.toList(), Randomizer.current()); + return makeSelection(inputStream, Randomizer.current()); } public static Optional makeSelection(final Stream> inputStream, IRandomizer randomizer) { return makeSelection(inputStream.toList(), randomizer); } - public static Optional makeSelection(final Collection> selections) { - return makeSelection(selections, Randomizer.current()); - } - - public static Optional makeSelection(final Collection> selections, IRandomizer randomizer) { + public static Optional makeSelection(final List> selections, IRandomizer randomizer) { if (selections.isEmpty()) return Optional.empty(); - if (selections.size() == 1) { - T theItem; - if (selections instanceof List> theList) - theItem = theList.getFirst().getItem(); - else if (selections instanceof ObjectArray> theArray) - theItem = theArray.getFirst().getItem(); - else - theItem = selections.iterator().next().getItem(); - return Optional.of(theItem); - } - - int totalWeight = 0; - for (var e : selections) - totalWeight += e.getWeight(); - if (totalWeight == 0) - return Optional.empty(); - - int targetWeight = randomizer.nextInt(totalWeight); - - for (var e : selections) { - targetWeight -= e.getWeight(); - if (targetWeight < 0) - return Optional.of(e.getItem()); - } + if (selections.size() == 1) + return Optional.of(selections.getFirst().data()); - // Shouldn't get here - throw new RuntimeException("Bad weight table - ran off the end"); + return WeightedRandom.getRandomItem(randomizer, selections).map(IItem::data); } - public interface IItem { - - int getWeight(); - - T getItem(); + public interface IItem extends WeightedEntry { + T data(); } } \ No newline at end of file diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/collections/ObjectArray.java b/common/src/main/java/org/orecruncher/dsurround/lib/collections/ObjectArray.java index c597a1d2..48f91b6f 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/collections/ObjectArray.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/collections/ObjectArray.java @@ -62,6 +62,10 @@ public T getFirst() { return this.get(0); } + public T getLast() { + return this.get(this.size() - 1); + } + private void remove0(final int idx) { final Object m = this.data[--this.insertionIdx]; this.data[this.insertionIdx] = null; @@ -71,11 +75,11 @@ private void remove0(final int idx) { @SuppressWarnings("unchecked") @Override - public boolean removeIf(final Predicate pred) { + public boolean removeIf(@NotNull final Predicate filter) { boolean result = false; for (int i = this.insertionIdx - 1; i >= 0; i--) { final T t = (T) this.data[i]; - if (pred.test(t)) { + if (filter.test(t)) { result = true; this.remove0(i); } @@ -117,7 +121,7 @@ public boolean contains(final Object o) { return result; } - @SuppressWarnings({"unchecked", "hiding"}) + @SuppressWarnings({"unchecked", "TypeParameterHidesVisibleType"}) @Override public T @NotNull [] toArray(final T[] a) { // From ArrayList impl @@ -173,7 +177,7 @@ public boolean removeAll(final Collection c) { } @Override - public boolean retainAll(final Collection c) { + public boolean retainAll(@NotNull final Collection c) { return this.removeIf(entry -> !c.contains(entry)); } diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/gui/ColorPalette.java b/common/src/main/java/org/orecruncher/dsurround/lib/gui/ColorPalette.java index 9856d0cb..f2298414 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/gui/ColorPalette.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/gui/ColorPalette.java @@ -2,6 +2,7 @@ import net.minecraft.ChatFormatting; import net.minecraft.network.chat.TextColor; +import net.minecraft.util.FastColor; import net.minecraft.util.Mth; @SuppressWarnings("unused") @@ -74,20 +75,21 @@ public final class ColorPalette { public static final TextColor ANTIQUE_WHITE = of(250,235,215); public static final TextColor PEARLY_PURPLE = of(183,104,162); public static final TextColor FRESH_AIR = of(166,231,255); + public static final TextColor HOT_PINK = of("#ff69b4"); public static final TextColor LEMON = of(254, 251, 1); public static final TextColor ELECTRIC_GREEN = of(0, 237, 1); public static int getRed(int rgb) { - return (rgb >> 16) & 0xFF; + return FastColor.ARGB32.red(rgb); } public static int getGreen(int rgb) { - return (rgb >> 8) & 0xFF; + return FastColor.ARGB32.green(rgb); } public static int getBlue(int rgb) { - return rgb & 0xFF; + return FastColor.ARGB32.blue(rgb); } private static TextColor of(ChatFormatting formatColor) { @@ -98,14 +100,16 @@ private static TextColor of(String formatColor) { return TextColor.parseColor(formatColor).getOrThrow(); } + private static TextColor of(int rgb) { + return TextColor.fromRgb(rgb); + } + private static TextColor of(int red, int green, int blue) { - return TextColor.fromRgb(toRGB(red, green, blue)); + return of(toRGB(red, green, blue)); } static int toRGB(int red, int green, int blue) { - return ((red & 0xFF) << 16) | - ((green & 0xFF) << 8) | - ((blue & 0xFF)); + return FastColor.ARGB32.color(red, green, blue); } public static TextColor lerp(float scale, TextColor start, TextColor end) { @@ -116,9 +120,9 @@ public static TextColor lerp(float scale, TextColor start, TextColor end) { var endGreen = getGreen(end.getValue()); var endBlue = getBlue(end.getValue()); - var red = (int)Mth.lerp(scale, startRed, endRed); - var green = (int)Mth.lerp(scale, startGreen, endGreen); - var blue = (int)Mth.lerp(scale, startBlue, endBlue); + var red = Mth.lerpInt(scale, startRed, endRed); + var green = Mth.lerpInt(scale, startGreen, endGreen); + var blue = Mth.lerpInt(scale, startBlue, endBlue); return of(red, green, blue); } diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/ISeasonalInformation.java b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/ISeasonalInformation.java index b2dd3aa2..f04cf1ba 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/ISeasonalInformation.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/ISeasonalInformation.java @@ -1,10 +1,11 @@ package org.orecruncher.dsurround.lib.seasons; +import net.minecraft.client.multiplayer.ClientLevel; import net.minecraft.core.BlockPos; import net.minecraft.network.chat.Component; -import net.minecraft.world.level.Level; import net.minecraft.world.level.biome.Biome; import net.minecraft.world.level.levelgen.Heightmap; +import org.orecruncher.dsurround.lib.GameUtils; import java.util.Optional; @@ -18,95 +19,99 @@ public interface ISeasonalInformation { /** * Gets the name of the current season, if any */ - Optional getCurrentSeason(Level world); + Optional getCurrentSeason(); /** * Gets the translated season name from the provider resources. */ - Optional getCurrentSeasonTranslated(Level world); + Optional getCurrentSeasonTranslated(); /** * Indicates if the current season is considered Spring. */ - default boolean isSpring(Level world) { + default boolean isSpring() { return true; } /** * Indicates if the current season is considered Summer. */ - default boolean isSummer(Level world) { + default boolean isSummer() { return false; } /** * Indicates if the current season is considered Autumn/Fall. */ - default boolean isAutumn(Level world) { + default boolean isAutumn() { return false; } /** * Indicates if the current season is considered Winter. */ - default boolean isWinter(Level world) { + default boolean isWinter() { return false; } /** * Gets the temperature at the specified block location taking into account any seasonal variance. */ - float getTemperature(Level world, BlockPos blockPos); + float getTemperature(BlockPos blockPos); /** * Indicates whether the temperature at the given position is considered cold. For example, if the temp * is cold, the frost breath effect can be produced. */ - default boolean isColdTemperature(Level world, BlockPos blockPos) { - return this.getTemperature(world, blockPos) < 0.2F; + default boolean isColdTemperature(BlockPos blockPos) { + return this.getTemperature(blockPos) < 0.2F; } /** * Indicates whether the temperature at the given position is considered cold enough for snow. */ - default boolean isSnowTemperature(Level world, BlockPos blockPos) { - return this.getTemperature(world, blockPos) < 0.15F; + default boolean isSnowTemperature(BlockPos blockPos) { + return this.getTemperature(blockPos) < 0.15F; } /** * Gets the Y on the XZ plane at which precipitation will strike. */ - default int getPrecipitationHeight(Level world, BlockPos pos) { - return world.getHeight(Heightmap.Types.MOTION_BLOCKING, pos.getX(), pos.getZ()); + default int getPrecipitationHeight(BlockPos pos) { + return this.level().getHeight(Heightmap.Types.MOTION_BLOCKING, pos.getX(), pos.getZ()); } /** * Gets the possible precipitation that can occur in the biome at the specified position. */ - default Biome.Precipitation getPrecipitationAt(Level world, BlockPos blockPos) { - return world.getBiome(blockPos).value().getPrecipitationAt(blockPos); + default Biome.Precipitation getPrecipitationAt(BlockPos blockPos) { + return this.level().getBiome(blockPos).value().getPrecipitationAt(blockPos); } /** * Gets the active precipitation occurring at the specified position. */ - default Biome.Precipitation getActivePrecipitation(Level world, BlockPos pos) { - if (!world.isRaining()) { + default Biome.Precipitation getActivePrecipitation(BlockPos pos) { + if (!this.level().isRaining()) { // Not currently raining return Biome.Precipitation.NONE; } // If the biome has no rain... - if (this.getPrecipitationAt(world, pos) == Biome.Precipitation.NONE) + if (this.getPrecipitationAt(pos) == Biome.Precipitation.NONE) return Biome.Precipitation.NONE; // Is there a block above that is blocking the rainfall? - var p = this.getPrecipitationHeight(world, pos); + var p = this.getPrecipitationHeight(pos); if (p > pos.getY()) { return Biome.Precipitation.NONE; } // Use the temperature of the biome to get whether it is raining or snowing - return this.isSnowTemperature(world, pos) ? Biome.Precipitation.SNOW : Biome.Precipitation.RAIN; + return this.isSnowTemperature(pos) ? Biome.Precipitation.SNOW : Biome.Precipitation.RAIN; + } + + default ClientLevel level() { + return GameUtils.getWorld().orElseThrow(); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/SeasonManager.java b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/SeasonManager.java index 809ece41..4fd712c3 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/SeasonManager.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/SeasonManager.java @@ -3,6 +3,7 @@ import dev.architectury.platform.Platform; import org.orecruncher.dsurround.Constants; import org.orecruncher.dsurround.lib.Library; +import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.lib.seasons.compat.SereneSeasons; import org.orecruncher.dsurround.lib.seasons.compat.VanillaSeasons; @@ -12,9 +13,9 @@ public class SeasonManager { static { if (Platform.isModLoaded(Constants.SERENE_SEASONS)) { - HANDLER = new SereneSeasons(); + HANDLER = ContainerManager.resolve(SereneSeasons.class); } else { - HANDLER = new VanillaSeasons(); + HANDLER = ContainerManager.resolve(VanillaSeasons.class); } Library.LOGGER.info("Season provider: %s", HANDLER.getProviderName()); diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/SereneSeasons.java b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/SereneSeasons.java index 4b534e7b..bf298753 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/SereneSeasons.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/SereneSeasons.java @@ -1,9 +1,10 @@ package org.orecruncher.dsurround.lib.seasons.compat; +import net.minecraft.client.multiplayer.ClientLevel; import net.minecraft.core.BlockPos; import net.minecraft.network.chat.Component; -import net.minecraft.world.level.Level; import net.minecraft.world.level.biome.Biome; +import org.orecruncher.dsurround.config.libraries.IDimensionInformation; import sereneseasons.api.season.Season; import sereneseasons.api.season.SeasonHelper; import sereneseasons.season.SeasonHooks; @@ -18,20 +19,23 @@ public class SereneSeasons extends AbstractSeasonProvider { private Season.TropicalSeason tropicalSeason; private Component computed; - public SereneSeasons() { + private final IDimensionInformation dimensionInformation; + + public SereneSeasons(IDimensionInformation dimensionInformation) { super("Serene Seasons"); + this.dimensionInformation = dimensionInformation; } @Override - public Optional getCurrentSeason(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public Optional getCurrentSeason() { + var helper = SeasonHelper.getSeasonState(this.level()); var subSeason = helper.getSubSeason(); return Optional.of(Component.literal(subSeason.toString())); } @Override - public Optional getCurrentSeasonTranslated(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public Optional getCurrentSeasonTranslated() { + var helper = SeasonHelper.getSeasonState(this.level()); if (this.subSeason != helper.getSubSeason() || this.tropicalSeason != helper.getTropicalSeason()) { var subSeasonKey = "desc.sereneseasons." + helper.getSeason().toString().toLowerCase(Locale.ROOT); var tropicalSeasonKey = "desc.sereneseasons." + helper.getTropicalSeason().toString().toLowerCase(Locale.ROOT); @@ -45,35 +49,42 @@ public Optional getCurrentSeasonTranslated(Level world) { return Optional.of(this.computed); } - public boolean isSpring(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public boolean isSpring() { + var helper = SeasonHelper.getSeasonState(this.level()); return helper.getSeason() == Season.SPRING; } - public boolean isSummer(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public boolean isSummer() { + var helper = SeasonHelper.getSeasonState(this.level()); return helper.getSeason() == Season.SUMMER; } - public boolean isAutumn(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public boolean isAutumn() { + var helper = SeasonHelper.getSeasonState(this.level()); return helper.getSeason() == Season.AUTUMN; } - public boolean isWinter(Level world) { - var helper = SeasonHelper.getSeasonState(world); + public boolean isWinter() { + var helper = SeasonHelper.getSeasonState(this.level()); return helper.getSeason() == Season.WINTER; } @Override - public Biome.Precipitation getPrecipitationAt(Level world, BlockPos blockPos) { - var biome = world.getBiome(blockPos); - return SeasonHooks.getPrecipitationAtSeasonal(world, biome, blockPos); + public Biome.Precipitation getPrecipitationAt(BlockPos blockPos) { + var level = this.level(); + var biome = level.getBiome(blockPos); + return SeasonHooks.getPrecipitationAtSeasonal(level, biome, blockPos); + } + + @Override + public float getTemperature(BlockPos blockPos) { + var level = this.level(); + var biome = level.getBiome(blockPos); + return SeasonHooks.getBiomeTemperature(level, biome, blockPos); } @Override - public float getTemperature(Level world, BlockPos blockPos) { - var biome = world.getBiome(blockPos); - return SeasonHooks.getBiomeTemperature(world, biome, blockPos); + public ClientLevel level() { + return this.dimensionInformation.level(); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/VanillaSeasons.java b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/VanillaSeasons.java index ac7e38c6..a9b19d3e 100644 --- a/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/VanillaSeasons.java +++ b/common/src/main/java/org/orecruncher/dsurround/lib/seasons/compat/VanillaSeasons.java @@ -2,7 +2,6 @@ import net.minecraft.core.BlockPos; import net.minecraft.network.chat.Component; -import net.minecraft.world.level.Level; import org.orecruncher.dsurround.mixinutils.IBiomeExtended; import java.util.Optional; @@ -14,18 +13,18 @@ public VanillaSeasons() { } @Override - public Optional getCurrentSeason(Level world) { + public Optional getCurrentSeason() { return Optional.of(Component.translatable("dsurround.text.seasons.spring")); } @Override - public Optional getCurrentSeasonTranslated(Level world) { - return getCurrentSeason(world); + public Optional getCurrentSeasonTranslated() { + return getCurrentSeason(); } @Override - public float getTemperature(Level world, BlockPos blockPos) { - var biome = world.getBiome(blockPos).value(); + public float getTemperature(BlockPos blockPos) { + var biome = this.level().getBiome(blockPos).value(); return ((IBiomeExtended)(Object)biome).dsurround_getTemperature(blockPos); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinMusicManager.java b/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinMusicManager.java index 1ed5f006..a1153acb 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinMusicManager.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinMusicManager.java @@ -2,9 +2,11 @@ import net.minecraft.client.resources.sounds.SoundInstance; import net.minecraft.client.sounds.MusicManager; +import net.minecraft.network.chat.Component; import net.minecraft.sounds.Music; import org.jetbrains.annotations.Nullable; import org.orecruncher.dsurround.gui.sound.SoundToast; +import org.orecruncher.dsurround.lib.gui.ColorPalette; import org.orecruncher.dsurround.mixinutils.IMusicManager; import org.orecruncher.dsurround.mixinutils.MixinHelpers; import org.spongepowered.asm.mixin.Mixin; @@ -64,6 +66,21 @@ public void dsurround_setPaused(boolean flag) { } } + @Override + public Component dsurround_whatsPlaying() { + if (this.currentMusic == null) + return Component.translatable("dsurround.text.musicmanager.nothing"); + + // Lookup meta information + var metaData = MixinHelpers.SOUND_LIBRARY.getSoundMetadata(this.currentMusic.getLocation()); + if (metaData == null || Component.empty().equals(metaData.getTitle())) + return Component.literal(this.currentMusic.getLocation().toString()); + + var title = metaData.getTitle().copy().withColor(ColorPalette.PUMPKIN_ORANGE.getValue()); + var author = metaData.getCredits().get(0).author().copy().withColor(ColorPalette.WHEAT.getValue()); + return Component.translatable("dsurround.text.musicmanager.playing", title, author, Component.translationArg(this.currentMusic.getLocation())); + } + @Inject(method = "tick()V", at = @At("HEAD"), cancellable = true) public void dsurround_pauseTickCheck(CallbackInfo ci) { if (this.dsurround_pauseTicking) diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinSoundEngine.java b/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinSoundEngine.java index 845933a0..ba981b62 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinSoundEngine.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixins/audio/MixinSoundEngine.java @@ -60,8 +60,12 @@ public void dsurround_onSoundPlay(SoundInstance soundInstance, CallbackInfo ci, @Inject(method = "play(Lnet/minecraft/client/resources/sounds/SoundInstance;)V", at = @At("HEAD"), cancellable = true) private void dsurround_play(SoundInstance sound, CallbackInfo ci) { try { + // Check to see if the sound is blocked or being culled if (SoundInstanceHandler.shouldBlockSoundPlay(sound)) ci.cancel(); + // Attempt a remapping if configured to do so + if (SoundInstanceHandler.remapSoundPlay(sound)) + ci.cancel(); } catch (final Exception t) { MixinHelpers.LOGGER.error(t, "Error in dsurround_play()!"); } diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinBiome.java b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinBiome.java index 2b95ca24..4e624bbd 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinBiome.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinBiome.java @@ -7,6 +7,7 @@ import org.orecruncher.dsurround.config.biome.BiomeInfo; import org.orecruncher.dsurround.lib.random.Randomizer; import org.orecruncher.dsurround.mixinutils.IBiomeExtended; +import org.orecruncher.dsurround.mixinutils.MixinHelpers; import org.spongepowered.asm.mixin.Final; import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.Shadow; @@ -56,16 +57,18 @@ public Biome.ClimateSettings dsurround_getWeather() { public abstract float dsurround_getTemperature(BlockPos pos); /** - * Obtain fog color from Dynamic Surroundings' config if available. + * Get fog color from Dynamic Surroundings' config if available. * * @param cir Mixin callback result */ @Inject(method = "getFogColor()I", at = @At("HEAD"), cancellable = true) public void dsurround_getFogColor(CallbackInfoReturnable cir) { - if (this.dsurround_info != null) { - var color = this.dsurround_info.getFogColor(); - if (color != null) - cir.setReturnValue(color.getValue()); + if (MixinHelpers.fogOptions.enableFogEffects && MixinHelpers.fogOptions.enableBiomeFog) { + if (this.dsurround_info != null) { + var color = this.dsurround_info.getFogColor(); + if (color != null) + cir.setReturnValue(color.getValue()); + } } } diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinFogRenderer.java b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinFogRenderer.java index faa673ff..5bd7fe13 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinFogRenderer.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinFogRenderer.java @@ -1,25 +1,34 @@ package org.orecruncher.dsurround.mixins.core; +import com.llamalad7.mixinextras.sugar.Local; +import com.mojang.blaze3d.systems.RenderSystem; import net.minecraft.client.Camera; import net.minecraft.client.renderer.FogRenderer; -import net.minecraft.world.entity.Entity; import net.minecraft.world.level.material.FogType; import org.orecruncher.dsurround.eventing.ClientEventHooks; import org.spongepowered.asm.mixin.Mixin; import org.spongepowered.asm.mixin.injection.At; import org.spongepowered.asm.mixin.injection.Inject; import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; -import org.spongepowered.asm.mixin.injection.callback.LocalCapture; @Mixin(FogRenderer.class) public class MixinFogRenderer { - @Inject(method = "setupFog(Lnet/minecraft/client/Camera;Lnet/minecraft/client/renderer/FogRenderer$FogMode;FZF)V", at = @At("RETURN"), locals = LocalCapture.CAPTURE_FAILHARD) - private static void dsurround_renderFog(Camera camera, FogRenderer.FogMode fogMode, float f, boolean bl, float g, CallbackInfo ci, FogType fogType, Entity entity, FogRenderer.FogData fogData) { + @Inject(method = "setupFog(Lnet/minecraft/client/Camera;Lnet/minecraft/client/renderer/FogRenderer$FogMode;FZF)V", at = @At("RETURN")) + private static void dsurround_renderFog(Camera camera, FogRenderer.FogMode fogMode, float f, boolean bl, float g, CallbackInfo ci, @Local FogType fogType, @Local FogRenderer.FogData fogData) { if (fogData.mode != FogRenderer.FogMode.FOG_TERRAIN || fogType != FogType.NONE) return; - ClientEventHooks.FOG_RENDER_EVENT.raise().onRenderFog(fogData, f, g); + // At this point, Minecraft has already configured fog. It's possible that another + // mixin fired and configured as well. We cannot trust the state of fogData, so + // we interrogate the shader directly to see what was configured. (Nostalgic Tweaks + // uses this approach.) + var data = new FogRenderer.FogData(fogData.mode); + data.start = RenderSystem.getShaderFogStart(); + data.end = RenderSystem.getShaderFogEnd(); + data.shape = RenderSystem.getShaderFogShape(); + + ClientEventHooks.FOG_RENDER_EVENT.raise().onRenderFog(data, f, g); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinIngameHud.java b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinIngameHud.java deleted file mode 100644 index fb3105b9..00000000 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinIngameHud.java +++ /dev/null @@ -1,30 +0,0 @@ -package org.orecruncher.dsurround.mixins.core; - -import net.minecraft.client.DeltaTracker; -import net.minecraft.client.Minecraft; -import net.minecraft.client.gui.Gui; -import net.minecraft.client.gui.GuiGraphics; -import org.orecruncher.dsurround.gui.overlay.OverlayManager; -import org.orecruncher.dsurround.lib.di.ContainerManager; -import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.Unique; -import org.spongepowered.asm.mixin.injection.At; -import org.spongepowered.asm.mixin.injection.Inject; -import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; - -@Mixin(Gui.class) -public class MixinIngameHud { - - @Unique - private OverlayManager dsurround_overlayManager; - - @Inject(method = "(Lnet/minecraft/client/Minecraft;)V", at = @At("RETURN")) - public void dsurround_constructor(Minecraft minecraftClient, CallbackInfo ci) { - this.dsurround_overlayManager = ContainerManager.resolve(OverlayManager.class); - } - - @Inject(method = "render", at = @At(value = "RETURN")) - public void dsurround_render(GuiGraphics guiGraphics, DeltaTracker deltaTracker, CallbackInfo ci) { - this.dsurround_overlayManager.render(guiGraphics, deltaTracker.getGameTimeDeltaTicks()); - } -} diff --git a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinParticleManager.java b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinParticleManager.java index d27c5888..dcb5d20c 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinParticleManager.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixins/core/MixinParticleManager.java @@ -2,28 +2,18 @@ import net.minecraft.client.particle.Particle; import net.minecraft.client.particle.ParticleEngine; -import net.minecraft.client.particle.ParticleRenderType; import net.minecraft.client.particle.SpriteSet; import net.minecraft.core.particles.ParticleOptions; import net.minecraft.resources.ResourceLocation; import org.spongepowered.asm.mixin.Mixin; -import org.spongepowered.asm.mixin.Mutable; import org.spongepowered.asm.mixin.gen.Accessor; import org.spongepowered.asm.mixin.gen.Invoker; -import java.util.List; import java.util.Map; @Mixin(ParticleEngine.class) public interface MixinParticleManager { - @Accessor("RENDER_ORDER") - static List dsurround_getParticleTextureSheets() { return null; } - - @Accessor("RENDER_ORDER") - @Mutable - static void dsurround_setParticleTextureSheets(List sheets) { } - @Accessor("spriteSets") Map dsurround_getSpriteAwareFactories(); diff --git a/common/src/main/java/org/orecruncher/dsurround/mixinutils/IMusicManager.java b/common/src/main/java/org/orecruncher/dsurround/mixinutils/IMusicManager.java index 59c83a78..bb46d418 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixinutils/IMusicManager.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixinutils/IMusicManager.java @@ -1,5 +1,7 @@ package org.orecruncher.dsurround.mixinutils; +import net.minecraft.network.chat.Component; + public interface IMusicManager { String dsurround_getDiagnosticText(); @@ -7,4 +9,6 @@ public interface IMusicManager { void dsurround_doCommand(String command); void dsurround_setPaused(boolean flag); + + Component dsurround_whatsPlaying(); } diff --git a/common/src/main/java/org/orecruncher/dsurround/mixinutils/MixinHelpers.java b/common/src/main/java/org/orecruncher/dsurround/mixinutils/MixinHelpers.java index c07066a5..68c29f8a 100644 --- a/common/src/main/java/org/orecruncher/dsurround/mixinutils/MixinHelpers.java +++ b/common/src/main/java/org/orecruncher/dsurround/mixinutils/MixinHelpers.java @@ -1,16 +1,23 @@ package org.orecruncher.dsurround.mixinutils; import org.orecruncher.dsurround.Configuration; +import org.orecruncher.dsurround.config.libraries.ISoundLibrary; import org.orecruncher.dsurround.config.libraries.ITagLibrary; import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.lib.logging.IModLog; +/** + * Static class definitions with mixins do not work well. Any statics have been placed + * here for mixin access. + */ public class MixinHelpers { public static final IModLog LOGGER = ContainerManager.resolve(IModLog.class); public static final ITagLibrary TAG_LIBRARY = ContainerManager.resolve(ITagLibrary.class); + public static final ISoundLibrary SOUND_LIBRARY = ContainerManager.resolve(ISoundLibrary.class); public static final Configuration.SoundSystem soundSystemConfig = ContainerManager.resolve(Configuration.SoundSystem.class); public static final Configuration.FootstepAccents footstepAccentsConfig = ContainerManager.resolve(Configuration.FootstepAccents.class); public static final Configuration.ParticleTweaks particleTweaksConfig = ContainerManager.resolve(Configuration.ParticleTweaks.class); public static final Configuration.SoundOptions soundOptions = ContainerManager.resolve(Configuration.SoundOptions.class); + public static final Configuration.FogOptions fogOptions = ContainerManager.resolve(Configuration.FogOptions.class); } diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/FogHandler.java b/common/src/main/java/org/orecruncher/dsurround/processing/FogHandler.java index e3ac1c19..4cdd3b1c 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/FogHandler.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/FogHandler.java @@ -41,11 +41,18 @@ private void renderFog(FogRenderer.FogData data, float renderDistance, float par RenderSystem.setShaderFogStart(this.lastData.start); RenderSystem.setShaderFogEnd(this.lastData.end); RenderSystem.setShaderFogShape(this.lastData.shape); + } else { + // Preserve for diagnostic trace even though action was not taken + this.lastData = data; } } @Override protected void gatherDiagnostics(CollectDiagnosticsEvent event) { - event.add(CollectDiagnosticsEvent.Section.Systems, "Fog: %f/%f, %s, %s".formatted(this.lastData.start, this.lastData.end, this.lastData.shape, this.lastData.mode)); + var text = "Fog: %f/%f, %s, %s ".formatted(this.lastData.start, this.lastData.end, this.lastData.shape, this.lastData.mode); + var disabledText = this.fogCalculator.getDisabledText(); + if (disabledText.isPresent()) + text += disabledText.get(); + event.add(CollectDiagnosticsEvent.Section.Systems, text); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/accents/FootstepAccents.java b/common/src/main/java/org/orecruncher/dsurround/processing/accents/FootstepAccents.java index d1a0446d..fe68ce63 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/accents/FootstepAccents.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/accents/FootstepAccents.java @@ -26,7 +26,6 @@ public FootstepAccents(Configuration config, IItemLibrary itemLibrary) { // Only register these providers if Presence Footsteps is not installed if (!Platform.isModLoaded(Constants.MOD_PRESENCE_FOOTSTEPS)) { this.providers.add(new WaterySurfaceAccent(config)); - this.providers.add(new LeavesAccent(config)); } } diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/accents/LeavesAccent.java b/common/src/main/java/org/orecruncher/dsurround/processing/accents/LeavesAccent.java deleted file mode 100644 index b9eb0c42..00000000 --- a/common/src/main/java/org/orecruncher/dsurround/processing/accents/LeavesAccent.java +++ /dev/null @@ -1,39 +0,0 @@ -package org.orecruncher.dsurround.processing.accents; - -import net.minecraft.core.BlockPos; -import net.minecraft.resources.ResourceLocation; -import net.minecraft.world.entity.LivingEntity; -import net.minecraft.world.level.block.state.BlockState; -import org.orecruncher.dsurround.Configuration; -import org.orecruncher.dsurround.Constants; -import org.orecruncher.dsurround.lib.collections.ObjectArray; -import org.orecruncher.dsurround.sound.ISoundFactory; -import org.orecruncher.dsurround.tags.BlockEffectTags; - -public class LeavesAccent implements IFootstepAccentProvider { - - private static final ResourceLocation LEAVES_FACTORY = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "footstep/leaves"); - - private final Configuration config; - - LeavesAccent(Configuration config) { - this.config = config; - } - - @Override - public void collect(LivingEntity entity, BlockPos pos, BlockState state, boolean isWaterLogged, ObjectArray acoustics) { - // Check for waterlogged. Don't want to squeak if waterlogged. - if (isWaterLogged) - return; - - if (FootstepAccents.TAG_LIBRARY.is(BlockEffectTags.LEAVES_STEP, state)) { - SOUND_LIBRARY.getSoundFactory(LEAVES_FACTORY) - .ifPresent(acoustics::add); - } - } - - @Override - public boolean isEnabled() { - return this.config.footstepAccents.enableLeafAccents; - } -} \ No newline at end of file diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/fog/BiomeFogRangeCalculator.java b/common/src/main/java/org/orecruncher/dsurround/processing/fog/BiomeFogRangeCalculator.java index 36f5ec1c..40732bf3 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/fog/BiomeFogRangeCalculator.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/fog/BiomeFogRangeCalculator.java @@ -23,7 +23,7 @@ public class BiomeFogRangeCalculator extends VanillaFogRangeCalculator { private float targetScale; public BiomeFogRangeCalculator(IBiomeLibrary biomeLibrary, Configuration.FogOptions fogOptions) { - super("BiomeFogRangeCalculator", fogOptions); + super("Biome", fogOptions); this.biomeLibrary = biomeLibrary; this.activeScale = this.targetScale = 0F; this.lastBlockPos = BlockPos.ZERO; diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/fog/HolisticFogRangeCalculator.java b/common/src/main/java/org/orecruncher/dsurround/processing/fog/HolisticFogRangeCalculator.java index 61fbb957..d855cccd 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/fog/HolisticFogRangeCalculator.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/fog/HolisticFogRangeCalculator.java @@ -11,6 +11,9 @@ import org.orecruncher.dsurround.lib.logging.ModLog; import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; +import java.util.Optional; +import java.util.stream.Collectors; + public class HolisticFogRangeCalculator implements IFogRangeCalculator { protected final IModLog logger; @@ -37,13 +40,13 @@ public String getName() { @Override public boolean enabled() { - return true; + return this.fogOptions.enableFogEffects; } @NotNull public FogRenderer.FogData render(@NotNull final FogRenderer.FogData data, float renderDistance, float partialTick) { - if (!this.fogOptions.enableFogEffects) + if (!this.enabled()) return data; float start = data.start; @@ -78,4 +81,18 @@ public void tick() { public void disconnect() { this.calculators.forEach(IFogRangeCalculator::disconnect); } + + public Optional getDisabledText() { + if (!this.enabled()) { + return Optional.of("(ALL DISABLED)"); + } + + var result = this.calculators.stream().filter(e -> !e.enabled()).map(IFogRangeCalculator::getName).collect(Collectors.joining(", ")); + + if (result.isEmpty()) { + return Optional.empty(); + } + + return Optional.of("(DISABLED: " + result + ")"); + } } diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/fog/MorningFogRangeCalculator.java b/common/src/main/java/org/orecruncher/dsurround/processing/fog/MorningFogRangeCalculator.java index 51db66ce..e57d8b50 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/fog/MorningFogRangeCalculator.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/fog/MorningFogRangeCalculator.java @@ -2,50 +2,38 @@ import net.minecraft.client.renderer.FogRenderer; import net.minecraft.util.Mth; +import net.minecraft.util.random.SimpleWeightedRandomList; import org.jetbrains.annotations.NotNull; import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.lib.GameUtils; import org.orecruncher.dsurround.lib.MinecraftClock; +import org.orecruncher.dsurround.lib.random.Randomizer; import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; -import org.orecruncher.dsurround.lib.WeightTable; - -import java.util.List; public class MorningFogRangeCalculator extends VanillaFogRangeCalculator { - private record FogDensityEntry(int weight, FogDensity density) implements WeightTable.IItem { - @Override - public int getWeight() { - return this.weight(); - } - @Override - public FogDensity getItem() { - return this.density(); - } - } - - private static final FogDensityEntry[] SPRING_FOG = { - new FogDensityEntry(30, FogDensity.NORMAL), - new FogDensityEntry(20, FogDensity.MEDIUM), - new FogDensityEntry(10, FogDensity.HEAVY) - }; - - private static final FogDensityEntry[] SUMMER_FOG = { - new FogDensityEntry(20, FogDensity.LIGHT), - new FogDensityEntry(10, FogDensity.NONE) - }; - - private static final FogDensityEntry[] AUTUMN_FOG = { - new FogDensityEntry(10, FogDensity.NORMAL), - new FogDensityEntry(20, FogDensity.MEDIUM), - new FogDensityEntry(10, FogDensity.HEAVY) - }; - - private static final FogDensityEntry[] WINTER_FOG = { - new FogDensityEntry(20, FogDensity.LIGHT), - new FogDensityEntry(20, FogDensity.NORMAL), - new FogDensityEntry(10, FogDensity.MEDIUM) - }; + private static final SimpleWeightedRandomList SPRING_FOG = new SimpleWeightedRandomList.Builder() + .add(FogDensity.NORMAL, 30) + .add(FogDensity.MEDIUM, 20) + .add(FogDensity.HEAVY, 10) + .build(); + + private static final SimpleWeightedRandomList SUMMER_FOG = new SimpleWeightedRandomList.Builder() + .add(FogDensity.LIGHT, 20) + .add(FogDensity.NONE, 10) + .build(); + + private static final SimpleWeightedRandomList AUTUMN_FOG = new SimpleWeightedRandomList.Builder() + .add(FogDensity.NORMAL, 10) + .add(FogDensity.MEDIUM, 20) + .add(FogDensity.HEAVY, 10) + .build(); + + private static final SimpleWeightedRandomList WINTER_FOG = new SimpleWeightedRandomList.Builder() + .add(FogDensity.LIGHT, 20) + .add(FogDensity.NORMAL, 20) + .add(FogDensity.MEDIUM, 10) + .build(); protected final ISeasonalInformation seasonInfo; protected final MinecraftClock clock; @@ -53,7 +41,7 @@ public FogDensity getItem() { protected FogDensity type = FogDensity.NONE; public MorningFogRangeCalculator(ISeasonalInformation seasonInfo, Configuration.FogOptions fogOptions) { - super("MorningFogRangeCalculator", fogOptions); + super("Morning", fogOptions); this.seasonInfo = seasonInfo; this.clock = new MinecraftClock(); } @@ -112,20 +100,19 @@ private float getCelestialAngleDegrees() { @NotNull protected FogDensity getFogType() { - FogDensityEntry[] selections; - var clientLevel = GameUtils.getWorld().orElseThrow(); - if (this.seasonInfo.isSpring(clientLevel)) + SimpleWeightedRandomList selections; + if (this.seasonInfo.isSpring()) selections = SPRING_FOG; - else if (this.seasonInfo.isSummer(clientLevel)) + else if (this.seasonInfo.isSummer()) selections = SUMMER_FOG; - else if (this.seasonInfo.isAutumn(clientLevel)) + else if (this.seasonInfo.isAutumn()) selections = AUTUMN_FOG; - else if (this.seasonInfo.isWinter(clientLevel)) + else if (this.seasonInfo.isWinter()) selections = WINTER_FOG; else // Shouldn't get here, but... return FogDensity.NONE; - return WeightTable.makeSelection(List.of(selections)).orElseThrow(); + return selections.getRandomValue(Randomizer.current()).orElseThrow(); } } \ No newline at end of file diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/fog/WeatherFogRangeCalculator.java b/common/src/main/java/org/orecruncher/dsurround/processing/fog/WeatherFogRangeCalculator.java index bb788980..952cbfda 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/fog/WeatherFogRangeCalculator.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/fog/WeatherFogRangeCalculator.java @@ -11,7 +11,7 @@ public class WeatherFogRangeCalculator extends VanillaFogRangeCalculator { protected static final float END_IMPACT = 0.4F; protected WeatherFogRangeCalculator(Configuration.FogOptions fogOptions) { - super("WeatherFogRangeCalculator", fogOptions); + super("Weather", fogOptions); } @Override diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/BiomeScanner.java b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/BiomeScanner.java index 13feea95..588aed08 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/BiomeScanner.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/BiomeScanner.java @@ -2,14 +2,14 @@ import it.unimi.dsi.fastutil.objects.Reference2IntOpenHashMap; import net.minecraft.core.BlockPos; +import net.minecraft.resources.ResourceLocation; import net.minecraft.tags.FluidTags; import net.minecraft.world.level.biome.Biome; import net.minecraft.world.level.biome.BiomeManager; import org.orecruncher.dsurround.config.libraries.IBiomeLibrary; -import org.orecruncher.dsurround.config.libraries.IDimensionLibrary; +import org.orecruncher.dsurround.config.libraries.IDimensionInformation; import org.orecruncher.dsurround.config.SyntheticBiome; import org.orecruncher.dsurround.config.biome.BiomeInfo; -import org.orecruncher.dsurround.config.dimension.DimensionInfo; import org.orecruncher.dsurround.lib.GameUtils; public final class BiomeScanner extends AbstractScanner { @@ -22,7 +22,7 @@ public final class BiomeScanner extends AbstractScanner { private static final int SURVEY_VERTICAL_OFFSET = SURVEY_VERTICAL_DIMENSION / 4 - 1; private static final int MAX_SURVEY_VOLUME = SURVEY_HORIZONTAL_DIMENSION * SURVEY_HORIZONTAL_DIMENSION * SURVEY_VERTICAL_DIMENSION; - private DimensionInfo surveyedDimension; + private ResourceLocation surveyedDimension; private boolean isUnderWater; private BiomeInfo logicalBiomeInfo; @@ -33,12 +33,12 @@ public final class BiomeScanner extends AbstractScanner { private Biome surveyedBiome = null; private BlockPos surveyedPosition = BlockPos.ZERO; private final IBiomeLibrary biomeLibrary; - private final IDimensionLibrary dimensionLibrary; + private final IDimensionInformation dimensionInformation; private final CeilingScanner ceilingScanner; - public BiomeScanner(IBiomeLibrary biomeLibrary, IDimensionLibrary dimensionLibrary, CeilingScanner ceilingScanner) { + public BiomeScanner(IBiomeLibrary biomeLibrary, IDimensionInformation dimensionInformation, CeilingScanner ceilingScanner) { this.biomeLibrary = biomeLibrary; - this.dimensionLibrary = dimensionLibrary; + this.dimensionInformation = dimensionInformation; this.ceilingScanner = ceilingScanner; } @@ -46,7 +46,7 @@ public BiomeInfo playerLogicBiomeInfo() { return this.logicalBiomeInfo; } - public DimensionInfo getDimInfo() { + public ResourceLocation getDimInfo() { return this.surveyedDimension; } @@ -63,17 +63,16 @@ public void tick(long tickCount) { var world = player.level(); var position = player.blockPosition(); - var dimensionInfo = this.dimensionLibrary.getData(world); var biomes = world.getBiomeManager(); var playerBiome = biomes.getBiome(position); if (this.surveyedBiome != playerBiome.value() - || !this.surveyedDimension.equals(dimensionInfo) + || !this.surveyedDimension.equals(this.dimensionInformation.name()) || !this.surveyedPosition.equals(position)) { this.surveyedBiome = playerBiome.value(); this.surveyedPosition = position; - this.surveyedDimension = dimensionInfo; + this.surveyedDimension = this.dimensionInformation.name(); this.weights = new Reference2IntOpenHashMap<>(8); @@ -97,7 +96,7 @@ else if (playerBiomeInfo.isOcean()) return; } - this.logicalBiomeInfo = this.resolveBiome(dimensionInfo, biomes, position); + this.logicalBiomeInfo = this.resolveBiome(biomes, position); for (int z = 0; z < SURVEY_HORIZONTAL_DIMENSION; z++) { var dZ = z - SURVEY_HORIZONTAL_OFFSET + this.surveyedPosition.getZ(); @@ -108,7 +107,7 @@ else if (playerBiomeInfo.isOcean()) for (int y = 0; y < SURVEY_VERTICAL_DIMENSION; y++) { var dY = y - SURVEY_VERTICAL_OFFSET + this.surveyedPosition.getY(); this.mutable.setY(dY); - var info = this.resolveBiome(dimensionInfo, biomes, this.mutable); + var info = this.resolveBiome(biomes, this.mutable); this.weights.addTo(info, 1); } } @@ -117,7 +116,7 @@ else if (playerBiomeInfo.isOcean()) } } - private BiomeInfo resolveBiome(DimensionInfo dimInfo, BiomeManager access, BlockPos pos) { + private BiomeInfo resolveBiome(BiomeManager access, BlockPos pos) { // Get the biome at the specified position var biome = access.getBiome(pos).value(); @@ -130,24 +129,24 @@ private BiomeInfo resolveBiome(DimensionInfo dimInfo, BiomeManager access, Block // Are we simulating underground? This is done by checking the Y level. var y = pos.getY(); - if (y < (dimInfo.getSeaLevel() - UNDERGROUND_THRESHOLD_OFFSET)) { + if (y < (this.dimensionInformation.seaLevel() - UNDERGROUND_THRESHOLD_OFFSET)) { return this.biomeLibrary.getBiomeInfo(SyntheticBiome.UNDERGROUND); } - // Inside check overrides everything else. Some dimensions are always out side + // Inside check overrides everything else. Some dimensions are always outside, // so those are excluded (like the nether). - if (!dimInfo.alwaysOutside() && this.ceilingScanner.isReallyInside()) { + if (!this.dimensionInformation.alwaysOutside() && this.ceilingScanner.isReallyInside()) { // If it's not underground, and we are inside, return INSIDE return this.biomeLibrary.getBiomeInfo(SyntheticBiome.INSIDE); } // Are we really up in the sky - if (y >= dimInfo.getSpaceHeight()) { + if (y >= this.dimensionInformation.getSpaceHeight()) { return this.biomeLibrary.getBiomeInfo(SyntheticBiome.SPACE); } // Up in the clouds - if (y >= dimInfo.getCloudHeight()) { + if (y >= this.dimensionInformation.getCloudHeight()) { return this.biomeLibrary.getBiomeInfo(SyntheticBiome.CLOUDS); } diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/CeilingScanner.java b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/CeilingScanner.java index 92f41293..f86cb754 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/CeilingScanner.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/CeilingScanner.java @@ -8,7 +8,7 @@ import net.minecraft.world.level.block.Block; import net.minecraft.world.level.block.state.BlockState; import org.orecruncher.dsurround.config.libraries.IDimensionLibrary; -import org.orecruncher.dsurround.config.dimension.DimensionInfo; +import org.orecruncher.dsurround.config.DimensionInfo; import org.orecruncher.dsurround.config.libraries.ITagLibrary; import org.orecruncher.dsurround.lib.GameUtils; import org.orecruncher.dsurround.lib.collections.ObjectArray; diff --git a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/SystemsScanner.java b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/SystemsScanner.java index 96852c7c..ae096fd1 100644 --- a/common/src/main/java/org/orecruncher/dsurround/processing/scanner/SystemsScanner.java +++ b/common/src/main/java/org/orecruncher/dsurround/processing/scanner/SystemsScanner.java @@ -1,5 +1,6 @@ package org.orecruncher.dsurround.processing.scanner; +import net.minecraft.core.BlockBox; import net.minecraft.core.BlockPos; import net.minecraft.network.chat.Component; import net.minecraft.world.level.Level; @@ -7,7 +8,6 @@ import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.effects.IBlockEffect; import org.orecruncher.dsurround.effects.IEffectSystem; -import org.orecruncher.dsurround.lib.BlockPosUtil; import org.orecruncher.dsurround.lib.GameUtils; import org.orecruncher.dsurround.lib.collections.ObjectArray; import org.orecruncher.dsurround.lib.di.Cacheable; @@ -64,26 +64,25 @@ public void tick() { final boolean sittingStill = this.lastPos.equals(current); this.lastPos = current; - Predicate pred; + Predicate filter; if (!sittingStill) { var range = this.config.blockEffects.blockEffectRange; - var minPoint = current.offset(-range, -range, -range); - var maxPoint = current.offset(range, range, range); + var blockBox = BlockBox.of(current.offset(-range, -range, -range), current.offset(range, range, range)); - pred = system -> { - if (!BlockPosUtil.contains(system.getPos(), minPoint, maxPoint)) { - system.remove(); - } else { + filter = system -> { + if (blockBox.contains(system.getPos())) { system.tick(); + } else { + system.remove(); } return system.isDone(); }; } else { - pred = EFFECT_PREDICATE; + filter = EFFECT_PREDICATE; } - this.processIfEnabled(true, system -> system.tick(pred)); + this.processIfEnabled(true, system -> system.tick(filter)); } protected void processIfEnabled(boolean clearSystems, Consumer systemConsumer) { diff --git a/common/src/main/java/org/orecruncher/dsurround/runtime/audio/SoundFXUtils.java b/common/src/main/java/org/orecruncher/dsurround/runtime/audio/SoundFXUtils.java index 278ac24e..abeda9aa 100644 --- a/common/src/main/java/org/orecruncher/dsurround/runtime/audio/SoundFXUtils.java +++ b/common/src/main/java/org/orecruncher/dsurround/runtime/audio/SoundFXUtils.java @@ -20,7 +20,6 @@ import org.orecruncher.dsurround.lib.math.ReusableRaycastContext; import org.orecruncher.dsurround.lib.math.ReusableRaycastIterator; import org.orecruncher.dsurround.lib.seasons.ISeasonalInformation; -import org.orecruncher.dsurround.lib.world.WorldUtils; import org.orecruncher.dsurround.runtime.audio.effects.Effects; import org.orecruncher.dsurround.runtime.audio.effects.LowPassData; import org.orecruncher.dsurround.runtime.audio.effects.SourcePropertyFloat; @@ -352,9 +351,9 @@ private static float calculateWeatherAbsorption(final WorldContext ctx, final Ve final BlockPos high = BlockPos.containing(pt2); // Determine the precipitation type at each point - final Biome.Precipitation rt1 = SEASONAL_INFORMATION.getActivePrecipitation(ctx.world, low); - final Biome.Precipitation rt2 = SEASONAL_INFORMATION.getActivePrecipitation(ctx.world, mid); - final Biome.Precipitation rt3 = SEASONAL_INFORMATION.getActivePrecipitation(ctx.world, high); + final Biome.Precipitation rt1 = SEASONAL_INFORMATION.getActivePrecipitation(low); + final Biome.Precipitation rt2 = SEASONAL_INFORMATION.getActivePrecipitation(mid); + final Biome.Precipitation rt3 = SEASONAL_INFORMATION.getActivePrecipitation(high); // Calculate the impact of weather on dampening float factor = calcFactor(rt1, 0.25F); diff --git a/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/SeasonVariables.java b/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/SeasonVariables.java index dea39ebf..8e78783b 100644 --- a/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/SeasonVariables.java +++ b/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/SeasonVariables.java @@ -28,11 +28,10 @@ public ISeasonVariables getInterface() { @Override public void update(IVariableAccess variableAccess) { if (GameUtils.isInGame()) { - var world = GameUtils.getWorld().orElseThrow(); - this.isSpring = this.seasonalInformation.isSpring(world); - this.isSummer = this.seasonalInformation.isSummer(world); - this.isAutumn = this.seasonalInformation.isAutumn(world); - this.isWinter = this.seasonalInformation.isWinter(world); + this.isSpring = this.seasonalInformation.isSpring(); + this.isSummer = this.seasonalInformation.isSummer(); + this.isAutumn = this.seasonalInformation.isAutumn(); + this.isWinter = this.seasonalInformation.isWinter(); } else { this.isSpring = false; this.isSummer = false; diff --git a/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/WeatherVariables.java b/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/WeatherVariables.java index 75d46bb5..dc78d7ef 100644 --- a/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/WeatherVariables.java +++ b/common/src/main/java/org/orecruncher/dsurround/runtime/sets/impl/WeatherVariables.java @@ -38,9 +38,9 @@ public void update(IVariableAccess variableAccess) { this.thunderIntensity = world.getThunderLevel(1F); this.isRaining = world.isRaining(); this.isThundering = world.isThundering(); - this.temperature = this.seasonalInformation.getTemperature(world, player.blockPosition()); - this.isFrosty = this.seasonalInformation.isColdTemperature(world, player.blockPosition()); - this.canWaterFreeze = this.seasonalInformation.isSnowTemperature(world, player.blockPosition()); + this.temperature = this.seasonalInformation.getTemperature(player.blockPosition()); + this.isFrosty = this.seasonalInformation.isColdTemperature(player.blockPosition()); + this.canWaterFreeze = this.seasonalInformation.isSnowTemperature(player.blockPosition()); } else { this.rainIntensity = 0F; this.thunderIntensity = 0F; diff --git a/common/src/main/java/org/orecruncher/dsurround/sound/ISoundFactory.java b/common/src/main/java/org/orecruncher/dsurround/sound/ISoundFactory.java index 5b89d556..684c8e4b 100644 --- a/common/src/main/java/org/orecruncher/dsurround/sound/ISoundFactory.java +++ b/common/src/main/java/org/orecruncher/dsurround/sound/ISoundFactory.java @@ -9,6 +9,7 @@ import net.minecraft.world.phys.Vec3; import org.orecruncher.dsurround.lib.math.MathStuff; +@SuppressWarnings("unused") public interface ISoundFactory { /** @@ -89,7 +90,16 @@ default SimpleSoundInstance createAtLocation(Vec3 position) { * This sound instance is not attached. The properties of the sound instance are defined by the underlying * factory settings. */ - SimpleSoundInstance createAtLocation(Vec3 position, float volumeScale); + default SimpleSoundInstance createAtLocation(Vec3 position, float volumeScale) { + return this.createAtLocation(position.x(), position.y(), position.z(), volumeScale); + } + + /** + * Creates a sound instance at the specified location. The provided factor scales the sound volume. + * This sound instance is not attached. The properties of the sound instance are defined by the underlying + * factory settings. + */ + SimpleSoundInstance createAtLocation(double posX, double posY, double posZ, float volumeScale); /** * Creates a Music instance to be used with Minecraft's music manager diff --git a/common/src/main/java/org/orecruncher/dsurround/sound/SoundFactory.java b/common/src/main/java/org/orecruncher/dsurround/sound/SoundFactory.java index 2a3e45ba..4a0bd7a3 100644 --- a/common/src/main/java/org/orecruncher/dsurround/sound/SoundFactory.java +++ b/common/src/main/java/org/orecruncher/dsurround/sound/SoundFactory.java @@ -103,7 +103,7 @@ public EntityBoundSoundInstance attachToEntity(Entity entity) { } @Override - public SimpleSoundInstance createAtLocation(Vec3 position, float volumeScale) { + public SimpleSoundInstance createAtLocation(double posX, double posY, double posZ, float volumeScale) { return new SimpleSoundInstance( this.soundEvent.getLocation(), this.category, @@ -113,9 +113,9 @@ public SimpleSoundInstance createAtLocation(Vec3 position, float volumeScale) { this.isRepeatable, this.repeatDelay, this.attenuation, - position.x(), - position.y(), - position.z(), + posX, + posY, + posZ, this.global); } diff --git a/common/src/main/java/org/orecruncher/dsurround/sound/SoundInstanceHandler.java b/common/src/main/java/org/orecruncher/dsurround/sound/SoundInstanceHandler.java index b91554ca..cda8198b 100644 --- a/common/src/main/java/org/orecruncher/dsurround/sound/SoundInstanceHandler.java +++ b/common/src/main/java/org/orecruncher/dsurround/sound/SoundInstanceHandler.java @@ -5,9 +5,7 @@ import net.minecraft.client.resources.sounds.SoundInstance; import net.minecraft.resources.ResourceLocation; import net.minecraft.sounds.SoundSource; -import net.minecraft.sounds.SoundEvents; import net.minecraft.world.phys.Vec3; -import org.orecruncher.dsurround.Constants; import org.orecruncher.dsurround.Configuration; import org.orecruncher.dsurround.config.libraries.ISoundLibrary; import org.orecruncher.dsurround.gui.sound.ConfigSoundInstance; @@ -17,9 +15,7 @@ import org.orecruncher.dsurround.lib.di.ContainerManager; import org.orecruncher.dsurround.lib.threading.IClientTasking; -import java.util.HashSet; import java.util.Objects; -import java.util.Set; /** * Handles sound block and culling. @@ -31,15 +27,8 @@ public final class SoundInstanceHandler { private static final IAudioPlayer AUDIO_PLAYER = ContainerManager.resolve(IAudioPlayer.class); private static final ITickCount TICK_COUNT = ContainerManager.resolve(ITickCount.class); private static final Configuration.SoundSystem SOUND_SYSTEM_CONFIG = ContainerManager.resolve(Configuration.SoundSystem.class); - private static final Configuration.SoundOptions THUNDERSTORM_CONFIG = ContainerManager.resolve(Configuration.SoundOptions.class); private static final Object2LongOpenHashMap SOUND_CULL = new Object2LongOpenHashMap<>(32); - private static final Set THUNDER_SOUNDS = new HashSet<>(); - private static final ResourceLocation THUNDER_SOUND_FACTORY = ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "thunder"); - - static { - THUNDER_SOUNDS.add(SoundEvents.LIGHTNING_BOLT_THUNDER.getLocation()); - } private static boolean isSoundBlocked(final ResourceLocation id) { return SOUND_LIBRARY.isBlocked(id); @@ -79,20 +68,24 @@ public static boolean shouldBlockSoundPlay(final SoundInstance theSound) { return false; final ResourceLocation id = theSound.getLocation(); - - if (THUNDERSTORM_CONFIG.replaceThunderSounds && THUNDER_SOUNDS.contains(id)) { - // Yeah - a bit reentrant but it should be good - var soundFactory = SOUND_LIBRARY.getSoundFactory(THUNDER_SOUND_FACTORY); - if (soundFactory.isPresent()) { - var sound = soundFactory.get().createAtLocation(new Vec3(theSound.getX(), theSound.getY(), theSound.getZ())); - AUDIO_PLAYER.play(sound); - return true; - } - } - return isSoundBlocked(id) || isSoundCulledLogical(id); } + /** + * Remaps the sound to a different one and queues to the sound engine. + * + * @param theSound The sound that is being played + * @return True if the sound was remapped, false otherwise + */ + public static boolean remapSoundPlay(final SoundInstance theSound) { + return SOUND_LIBRARY.remapSound(theSound) + .map(s -> { + AUDIO_PLAYER.play(s); + return true; + }) + .orElse(false); + } + /** * Determines if a sound is in range of a listener based on the sound's properties. * diff --git a/common/src/main/java/org/orecruncher/dsurround/tags/ItemEffectTags.java b/common/src/main/java/org/orecruncher/dsurround/tags/ItemEffectTags.java index 6b0e6763..ccff63d3 100644 --- a/common/src/main/java/org/orecruncher/dsurround/tags/ItemEffectTags.java +++ b/common/src/main/java/org/orecruncher/dsurround/tags/ItemEffectTags.java @@ -22,6 +22,7 @@ public class ItemEffectTags { public static final TagKey SWORDS = of("swords"); public static final TagKey TOOLS = of("tools"); public static final TagKey COMPASSES = of("compasses"); + public static final TagKey COMPASS_WOBBLE = of("compass_wobble"); public static final TagKey CLOCKS = of("clocks"); private static TagKey of(String id) { diff --git a/common/src/main/resources/assets/biomesoplenty/dsconfigs/sound_mappings.json b/common/src/main/resources/assets/biomesoplenty/dsconfigs/sound_mappings.json new file mode 100644 index 00000000..1f41f66b --- /dev/null +++ b/common/src/main/resources/assets/biomesoplenty/dsconfigs/sound_mappings.json @@ -0,0 +1,13 @@ +[ + { + "soundEvent": "minecraft:block.wood.step", + "rules": [ + { + "blocks": [ + "biomesoplenty:glowshroom_block" + ], + "factory": "dsurround:footsteps.organic" + } + ] + } +] \ No newline at end of file diff --git a/common/src/main/resources/assets/biomesoplenty/dsconfigs/tags/block/effects/heat_producers.json b/common/src/main/resources/assets/biomesoplenty/dsconfigs/tags/block/effects/heat_producers.json new file mode 100644 index 00000000..24e7d408 --- /dev/null +++ b/common/src/main/resources/assets/biomesoplenty/dsconfigs/tags/block/effects/heat_producers.json @@ -0,0 +1,8 @@ +{ + "values": [ + { + "id": "biomesoplenty:brimstone_fumarole", + "required": false + } + ] +} \ No newline at end of file diff --git a/common/src/main/resources/assets/dsurround/dsconfigs/sound_factories.json b/common/src/main/resources/assets/dsurround/dsconfigs/sound_factories.json index 81d75575..e4df2735 100644 --- a/common/src/main/resources/assets/dsurround/dsconfigs/sound_factories.json +++ b/common/src/main/resources/assets/dsurround/dsconfigs/sound_factories.json @@ -322,5 +322,23 @@ "min": 1.0, "max": 1.2 } + }, + { + "location": "dsurround:footsteps/dirt_path", + "soundEvent" : "dsurround:footsteps.gravel", + "volume": 0.25, + "pitch": { + "min": 1.0, + "max": 1.2 + } + }, + { + "location": "dsurround:footsteps/ladder", + "soundEvent" : "dsurround:footsteps.bluntwood", + "volume": 0.5, + "pitch": { + "min": 1.3, + "max": 1.4 + } } ] \ No newline at end of file diff --git a/common/src/main/resources/assets/dsurround/dsconfigs/sound_mappings.json b/common/src/main/resources/assets/dsurround/dsconfigs/sound_mappings.json new file mode 100644 index 00000000..a4aefbc8 --- /dev/null +++ b/common/src/main/resources/assets/dsurround/dsconfigs/sound_mappings.json @@ -0,0 +1,157 @@ +[ + { + "soundEvent": "minecraft:entity.lightning_bolt.thunder", + "rules": [ + { + "factory": "dsurround:thunder" + } + ] + }, + { + "soundEvent": "minecraft:block.grass.step", + "rules": [ + { + "blocks": [ + "#dsurround:effects/leaves_step", + "minecraft:hay_block" + ], + "factory": "dsurround:footsteps.leaves_through" + }, + { + "blocks": [ + "minecraft:dirt_path" + ], + "factory": "dsurround:footsteps/dirt_path" + }, + { + "factory": "dsurround:footsteps.grass" + } + ] + }, + { + "soundEvent": "minecraft:block.gravel.step", + "rules": [ + { + "factory": "dsurround:footsteps.gravel" + } + ] + }, + { + "soundEvent": "minecraft:block.sand.step", + "rules": [ + { + "factory": "dsurround:footsteps.sand" + } + ] + }, + { + "soundEvent": "minecraft:block.dirt.step", + "rules": [ + { + "factory": "dsurround:footsteps.dirt" + } + ] + }, + { + "soundEvent": "minecraft:block.stone.step", + "rules": [ + { + "blocks": [ + "#minecraft:cauldrons" + ], + "factory": "dsurround:footsteps.metalbox" + }, + { + "factory": "dsurround:footsteps.stone" + } + ] + }, + { + "soundEvent": "minecraft:block.snow.step", + "rules": [ + { + "factory": "dsurround:footsteps.snow" + } + ] + }, + { + "soundEvent": "minecraft:block.glass.step", + "rules": [ + { + "blocks": [ + "#minecraft:ice", + "#c:ice" + ], + "factory": "dsurround:footsteps.muffledice" + }, + { + "factory": "dsurround:footsteps.wood" + } + ] + }, + { + "soundEvent": "minecraft:block.wood.step", + "rules": [ + { + "blocks": [ + "#minecraft:logs", + "#c:logs" + ], + "factory": "dsurround:footsteps.log" + }, + { + "blocks": [ + "#minecraft:fences", + "#c:fences" + ], + "factory": "dsurround:footsteps.bluntwood" + }, + { + "blocks": [ + "#minecraft:beehives", + "minecraft:red_mushroom_block", + "minecraft:brown_mushroom_block", + "minecraft:mushroom_stem", + "minecraft:pumpkin", + "minecraft:carved_pumpkin", + "minecraft:melon", + "minecraft:cocoa" + ], + "factory": "dsurround:footsteps.organic" + }, + { + "factory": "dsurround:footsteps.wood" + } + ] + }, + { + "soundEvent": "minecraft:block.metal.step", + "rules": [ + { + "blocks": [ + "minecraft:iron_bars" + ], + "factory": "dsurround:footsteps.metalbar" + }, + { + "factory": "dsurround:footsteps.metalbox" + } + ] + }, + { + "soundEvent": "minecraft:block.ladder.step", + "rules": [ + { + "factory": "dsurround:footsteps/ladder" + } + ] + }, + { + "soundEvent": "minecraft:block.wool.step", + "rules": [ + { + "factory": "dsurround:footsteps.rug" + } + ] + } +] \ No newline at end of file diff --git a/common/src/main/resources/assets/dsurround/dsconfigs/tags/item/effects/compass_wobble.json b/common/src/main/resources/assets/dsurround/dsconfigs/tags/item/effects/compass_wobble.json new file mode 100644 index 00000000..4e104c81 --- /dev/null +++ b/common/src/main/resources/assets/dsurround/dsconfigs/tags/item/effects/compass_wobble.json @@ -0,0 +1,5 @@ +{ + "values": [ + "minecraft:compass" + ] +} \ No newline at end of file diff --git a/common/src/main/resources/assets/dsurround/lang/en_us.json b/common/src/main/resources/assets/dsurround/lang/en_us.json index 3e07283c..431ac800 100644 --- a/common/src/main/resources/assets/dsurround/lang/en_us.json +++ b/common/src/main/resources/assets/dsurround/lang/en_us.json @@ -60,6 +60,8 @@ "dsurround.command.dsmm.pause.failure": "Unable to pause the Music Manager: %1$s", "dsurround.command.dsmm.unpause.success": "Music Manager was unpaused", "dsurround.command.dsmm.unpause.failure": "Unable to unpause the Music Manager: %1$s", + "dsurround.command.dsmm.whatsplaying.success": "\u00a72Playing\u00a7r: %1$s", + "dsurround.command.dsmm.whatsplaying.failure": "Unable to determine what the Music Manager is playing: %1$s", "dsurround.config.title": "Dynamic Surroundings Configuration Options", "dsurround.config.tooltip.clientRestartRequired": "CLIENT restart required", "dsurround.config.tooltip.worldRestartRequired": "WORLD restart required", @@ -92,6 +94,8 @@ "dsurround.config.soundOptions.ambientVolumeScaling.tooltip": "Ambient sounds played by the mod will be multiplied by this factor. The unit scale is 1/100, so a setting of 100 would multiply by 1. Volume calculations have several different factors, this setting being one of them. Regardless of these factors, the final volume will be restricted to a range of 0.0 - 1.0.", "dsurround.config.soundOptions.replaceThunderSounds": "Replace Vanilla thunder sounds", "dsurround.config.soundOptions.replaceThunderSounds.tooltip": "Enables replacement of thunder sounds with Dynamic Surroundings' version", + "dsurround.config.soundOptions.remapSounds": "Sound Remapping", + "dsurround.config.soundOptions.remapSounds.tooltip": "Enables sound remapping when sounds are played. Enabling tells Dynamic Surroundings to replace various step sounds with replacement versions.", "dsurround.config.soundOptions.allowScarySounds": "Allow playing scary sounds", "dsurround.config.soundOptions.allowScarySounds.tooltip": "Enables playing scary sounds. Scary can be subjective. It is up to mod pack authors to implement properly within the configuration. Disliked sounds also be blocked using the in-game sound configuration menu.", "dsurround.config.soundOptions.playBiomeMusicWhileCreative": "Play biome music while creative", @@ -128,6 +132,8 @@ "dsurround.config.entityEffects.enableBreathEffect.tooltip": "Enable/disable breath effect in cold biomes", "dsurround.config.entityEffects.enablePlayerToolbarEffect": "Player Toolbar Effects", "dsurround.config.entityEffects.enablePlayerToolbarEffect.tooltip": "Enable/disable player toolbar sound effects", + "dsurround.config.entityEffects.enableToolbarBlockSounds": "Player Toolbar Block Sounds", + "dsurround.config.entityEffects.enableToolbarBlockSounds.tooltip": "Enable/disable sound effects for blocks on the toolbar", "dsurround.config.entityEffects.enableSwingEffect": "Item Swing Effects", "dsurround.config.entityEffects.enableSwingEffect.tooltip": "Enable/disable item swing sound effects from players and mobs", "dsurround.config.entityEffects.enableBrushStepEffect": "Brush Step Effect", @@ -142,8 +148,6 @@ "dsurround.config.footstepAccents.enableWetSurfaceAccents.tooltip": "Enable/disable accents for when it is raining or blocks are waterlogged", "dsurround.config.footstepAccents.enableFloorSqueaks": "Floor Squeaks", "dsurround.config.footstepAccents.enableFloorSqueaks.tooltip": "Enable/disable accents for when the player is walking on squeaky blocks", - "dsurround.config.footstepAccents.enableLeafAccents": "Leafy Surface Accents", - "dsurround.config.footstepAccents.enableLeafAccents.tooltip": "Enable/disable accents for when the player is walking on leafy blocks", "dsurround.config.particleTweaks": "Particle Tweaks", "dsurround.config.particleTweaks.tooltip": "Configuration options for tweaking particle behavior", "dsurround.config.particleTweaks.suppressPlayerParticles": "Suppress Player Potion Effects", @@ -207,5 +211,7 @@ "dsurround.text.reloadassets": "[%1$s\u00a7r] \u00a7aReloaded configurations", "dsurround.text.seasons.spring": "Spring", "dsurround.text.toast.music.title": "Music \"%1$s\"", - "dsurround.text.toast.music.author": "by %1$s" + "dsurround.text.toast.music.author": "by %1$s", + "dsurround.text.musicmanager.nothing": "No music is playing", + "dsurround.text.musicmanager.playing": "%1$s by %2$s (%3$s)" } diff --git a/common/src/main/resources/assets/dsurround/lang/pl_pl.json b/common/src/main/resources/assets/dsurround/lang/pl_pl.json index 17eb732d..1a4b3b9d 100644 --- a/common/src/main/resources/assets/dsurround/lang/pl_pl.json +++ b/common/src/main/resources/assets/dsurround/lang/pl_pl.json @@ -137,8 +137,6 @@ "dsurround.config.footstepAccents.enableWetSurfaceAccents.tooltip": "Włącza/wyłącza akcenty dla deszczu or blocks are waterlogged", "dsurround.config.footstepAccents.enableFloorSqueaks": "Skrzypienie podłogi", "dsurround.config.footstepAccents.enableFloorSqueaks.tooltip": "Włącza/wyłącza akcenty dla gracza chodzącego po skrzypiących blokach", - "dsurround.config.footstepAccents.enableLeafAccents": "Akcenty liściastych powierzchni", - "dsurround.config.footstepAccents.enableLeafAccents.tooltip": "Włącza/wyłącza akcenty dla gracza chodzącego po liściastych blokach", "dsurround.config.particleTweaks": "Ulepszenia cząsteczek", "dsurround.config.particleTweaks.tooltip": "Opcje konfiguracji dla ulepszenia zachowania cząsteczek", "dsurround.config.particleTweaks.suppressPlayerParticles": "Wygaś efekty mikstur gracza", diff --git a/common/src/main/resources/assets/dsurround/lang/zh_cn.json b/common/src/main/resources/assets/dsurround/lang/zh_cn.json index dcfc4522..65d0edf3 100644 --- a/common/src/main/resources/assets/dsurround/lang/zh_cn.json +++ b/common/src/main/resources/assets/dsurround/lang/zh_cn.json @@ -137,8 +137,6 @@ "dsurround.config.footstepAccents.enableWetSurfaceAccents.tooltip": "启用/禁用 当下雨或方块被浸泡时的效果", "dsurround.config.footstepAccents.enableFloorSqueaks": "地板吱吱声", "dsurround.config.footstepAccents.enableFloorSqueaks.tooltip": "启用/禁用 当玩家走在吱吱作响的方块上的效果", - "dsurround.config.footstepAccents.enableLeafAccents": "叶子覆盖表面效果", - "dsurround.config.footstepAccents.enableLeafAccents.tooltip": "启用/禁用 当玩家行走在叶子方块上时的效果", "dsurround.config.particleTweaks": "粒子调整", "dsurround.config.particleTweaks.tooltip": "用于微调粒子行为的配置选项", "dsurround.config.particleTweaks.suppressPlayerParticles": "抑制玩家药水效果", diff --git a/common/src/main/resources/assets/dsurround/sounds.json b/common/src/main/resources/assets/dsurround/sounds.json index 35edc2d4..7e9e42eb 100644 --- a/common/src/main/resources/assets/dsurround/sounds.json +++ b/common/src/main/resources/assets/dsurround/sounds.json @@ -1219,6 +1219,245 @@ "dsurround:footsteps/leaves/leaves_through7" ] }, + "footsteps.bluntwood": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.bluntwood", + "sounds": [ + "dsurround:footsteps/bluntwood/bluntwood_walk1", + "dsurround:footsteps/bluntwood/bluntwood_walk10", + "dsurround:footsteps/bluntwood/bluntwood_walk11", + "dsurround:footsteps/bluntwood/bluntwood_walk2", + "dsurround:footsteps/bluntwood/bluntwood_walk3", + "dsurround:footsteps/bluntwood/bluntwood_walk4", + "dsurround:footsteps/bluntwood/bluntwood_walk5", + "dsurround:footsteps/bluntwood/bluntwood_walk6", + "dsurround:footsteps/bluntwood/bluntwood_walk7", + "dsurround:footsteps/bluntwood/bluntwood_walk8", + "dsurround:footsteps/bluntwood/bluntwood_walk9" + ] + }, + "footsteps.dirt": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.dirt", + "sounds": [ + "dsurround:footsteps/dirt/dirt_walk1", + "dsurround:footsteps/dirt/dirt_walk10", + "dsurround:footsteps/dirt/dirt_walk11", + "dsurround:footsteps/dirt/dirt_walk2", + "dsurround:footsteps/dirt/dirt_walk3", + "dsurround:footsteps/dirt/dirt_walk4", + "dsurround:footsteps/dirt/dirt_walk5", + "dsurround:footsteps/dirt/dirt_walk6", + "dsurround:footsteps/dirt/dirt_walk7", + "dsurround:footsteps/dirt/dirt_walk8", + "dsurround:footsteps/dirt/dirt_walk9" + ] + }, + "footsteps.grass": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.grass", + "sounds": [ + "dsurround:footsteps/grass/grass_walk1", + "dsurround:footsteps/grass/grass_walk10", + "dsurround:footsteps/grass/grass_walk2", + "dsurround:footsteps/grass/grass_walk3", + "dsurround:footsteps/grass/grass_walk4", + "dsurround:footsteps/grass/grass_walk5", + "dsurround:footsteps/grass/grass_walk6", + "dsurround:footsteps/grass/grass_walk7", + "dsurround:footsteps/grass/grass_walk8", + "dsurround:footsteps/grass/grass_walk9" + ] + }, + "footsteps.gravel": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.gravel", + "sounds": [ + "dsurround:footsteps/gravel/gravel_walk1", + "dsurround:footsteps/gravel/gravel_walk10", + "dsurround:footsteps/gravel/gravel_walk11", + "dsurround:footsteps/gravel/gravel_walk2", + "dsurround:footsteps/gravel/gravel_walk3", + "dsurround:footsteps/gravel/gravel_walk4", + "dsurround:footsteps/gravel/gravel_walk5", + "dsurround:footsteps/gravel/gravel_walk6", + "dsurround:footsteps/gravel/gravel_walk7", + "dsurround:footsteps/gravel/gravel_walk8", + "dsurround:footsteps/gravel/gravel_walk9" + ] + }, + "footsteps.mud": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.mud", + "sounds": [ + "dsurround:footsteps/mud/mud_walk1", + "dsurround:footsteps/mud/mud_walk2", + "dsurround:footsteps/mud/mud_walk3", + "dsurround:footsteps/mud/mud_walk4", + "dsurround:footsteps/mud/mud_walk5", + "dsurround:footsteps/mud/mud_walk6" + ] + }, + "footsteps.muffledice": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.muffledice", + "sounds": [ + "dsurround:footsteps/muffledice/muffledice_walk1", + "dsurround:footsteps/muffledice/muffledice_walk10", + "dsurround:footsteps/muffledice/muffledice_walk2", + "dsurround:footsteps/muffledice/muffledice_walk3", + "dsurround:footsteps/muffledice/muffledice_walk4", + "dsurround:footsteps/muffledice/muffledice_walk5", + "dsurround:footsteps/muffledice/muffledice_walk6", + "dsurround:footsteps/muffledice/muffledice_walk7", + "dsurround:footsteps/muffledice/muffledice_walk8", + "dsurround:footsteps/muffledice/muffledice_walk9" + ] + }, + "footsteps.organic": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.organic", + "sounds": [ + "dsurround:footsteps/organic/organic_walk1", + "dsurround:footsteps/organic/organic_walk2", + "dsurround:footsteps/organic/organic_walk3", + "dsurround:footsteps/organic/organic_walk4" + ] + }, + "footsteps.sand": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.sand", + "sounds": [ + "dsurround:footsteps/sand/sand_walk1", + "dsurround:footsteps/sand/sand_walk10", + "dsurround:footsteps/sand/sand_walk11", + "dsurround:footsteps/sand/sand_walk2", + "dsurround:footsteps/sand/sand_walk3", + "dsurround:footsteps/sand/sand_walk4", + "dsurround:footsteps/sand/sand_walk5", + "dsurround:footsteps/sand/sand_walk6", + "dsurround:footsteps/sand/sand_walk7", + "dsurround:footsteps/sand/sand_walk8", + "dsurround:footsteps/sand/sand_walk9" + ] + }, + "footsteps.snow": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.snow", + "sounds": [ + "dsurround:footsteps/snow/snow_walk1", + "dsurround:footsteps/snow/snow_walk10", + "dsurround:footsteps/snow/snow_walk11", + "dsurround:footsteps/snow/snow_walk2", + "dsurround:footsteps/snow/snow_walk3", + "dsurround:footsteps/snow/snow_walk4", + "dsurround:footsteps/snow/snow_walk5", + "dsurround:footsteps/snow/snow_walk6", + "dsurround:footsteps/snow/snow_walk7", + "dsurround:footsteps/snow/snow_walk8", + "dsurround:footsteps/snow/snow_walk9" + ] + }, + "footsteps.stone": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.stone", + "sounds": [ + "dsurround:footsteps/stone/stone_walk1", + "dsurround:footsteps/stone/stone_walk10", + "dsurround:footsteps/stone/stone_walk11", + "dsurround:footsteps/stone/stone_walk2", + "dsurround:footsteps/stone/stone_walk3", + "dsurround:footsteps/stone/stone_walk4", + "dsurround:footsteps/stone/stone_walk5", + "dsurround:footsteps/stone/stone_walk6", + "dsurround:footsteps/stone/stone_walk7", + "dsurround:footsteps/stone/stone_walk8", + "dsurround:footsteps/stone/stone_walk9" + ] + }, + "footsteps.wood": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.wood", + "sounds": [ + "dsurround:footsteps/wood/wood_walk1", + "dsurround:footsteps/wood/wood_walk10", + "dsurround:footsteps/wood/wood_walk11", + "dsurround:footsteps/wood/wood_walk2", + "dsurround:footsteps/wood/wood_walk3", + "dsurround:footsteps/wood/wood_walk4", + "dsurround:footsteps/wood/wood_walk5", + "dsurround:footsteps/wood/wood_walk6", + "dsurround:footsteps/wood/wood_walk7", + "dsurround:footsteps/wood/wood_walk8", + "dsurround:footsteps/wood/wood_walk9" + ] + }, + "footsteps.log": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.log", + "sounds": [ + "dsurround:footsteps/wood/log_walk1", + "dsurround:footsteps/wood/log_walk10", + "dsurround:footsteps/wood/log_walk11", + "dsurround:footsteps/wood/log_walk2", + "dsurround:footsteps/wood/log_walk3", + "dsurround:footsteps/wood/log_walk4", + "dsurround:footsteps/wood/log_walk5", + "dsurround:footsteps/wood/log_walk6", + "dsurround:footsteps/wood/log_walk7", + "dsurround:footsteps/wood/log_walk8", + "dsurround:footsteps/wood/log_walk9" + ] + }, + "footsteps.metalbox": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.metalbox", + "sounds": [ + "dsurround:footsteps/metalbox/metalbox_walk1", + "dsurround:footsteps/metalbox/metalbox_walk2", + "dsurround:footsteps/metalbox/metalbox_walk3", + "dsurround:footsteps/metalbox/metalbox_walk4", + "dsurround:footsteps/metalbox/metalbox_walk5", + "dsurround:footsteps/metalbox/metalbox_walk6", + "dsurround:footsteps/metalbox/metalbox_walk7", + "dsurround:footsteps/metalbox/metalbox_walk8", + "dsurround:footsteps/metalbox/metalbox_walk9" + ] + }, + "footsteps.metalbar": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.metalbar", + "sounds": [ + "dsurround:footsteps/metalbar/metalbar_walk1", + "dsurround:footsteps/metalbar/metalbar_walk10", + "dsurround:footsteps/metalbar/metalbar_walk11", + "dsurround:footsteps/metalbar/metalbar_walk2", + "dsurround:footsteps/metalbar/metalbar_walk3", + "dsurround:footsteps/metalbar/metalbar_walk4", + "dsurround:footsteps/metalbar/metalbar_walk5", + "dsurround:footsteps/metalbar/metalbar_walk6", + "dsurround:footsteps/metalbar/metalbar_walk7", + "dsurround:footsteps/metalbar/metalbar_walk8", + "dsurround:footsteps/metalbar/metalbar_walk9" + ] + }, + "footsteps.rug": { + "ds_category": "player", + "subtitle": "dsurround.sound.caption.footsteps.rug", + "sounds": [ + "dsurround:footsteps/rug/rug_walk1", + "dsurround:footsteps/rug/rug_walk10", + "dsurround:footsteps/rug/rug_walk11", + "dsurround:footsteps/rug/rug_walk2", + "dsurround:footsteps/rug/rug_walk3", + "dsurround:footsteps/rug/rug_walk4", + "dsurround:footsteps/rug/rug_walk5", + "dsurround:footsteps/rug/rug_walk6", + "dsurround:footsteps/rug/rug_walk7", + "dsurround:footsteps/rug/rug_walk8", + "dsurround:footsteps/rug/rug_walk9" + ] + }, "silence": { "sounds": [ "dsurround:ambient/silence" diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk1.ogg new file mode 100644 index 00000000..2b3df74d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk10.ogg new file mode 100644 index 00000000..0046ac85 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk11.ogg new file mode 100644 index 00000000..48f31e38 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk2.ogg new file mode 100644 index 00000000..5719bee5 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk3.ogg new file mode 100644 index 00000000..a695acc7 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk4.ogg new file mode 100644 index 00000000..e5cd889e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk5.ogg new file mode 100644 index 00000000..69b58a03 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk6.ogg new file mode 100644 index 00000000..63843423 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk7.ogg new file mode 100644 index 00000000..db28d3d9 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk8.ogg new file mode 100644 index 00000000..a2abb21e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk9.ogg new file mode 100644 index 00000000..fe8518f0 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/bluntwood/bluntwood_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk1.ogg new file mode 100644 index 00000000..386cb24d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk10.ogg new file mode 100644 index 00000000..a14e77f4 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk11.ogg new file mode 100644 index 00000000..38f1733d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk2.ogg new file mode 100644 index 00000000..da34cde6 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk3.ogg new file mode 100644 index 00000000..b17c93d6 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk4.ogg new file mode 100644 index 00000000..41e92f82 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk5.ogg new file mode 100644 index 00000000..f4b920c2 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk6.ogg new file mode 100644 index 00000000..2afab56e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk7.ogg new file mode 100644 index 00000000..78b8627c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk8.ogg new file mode 100644 index 00000000..ad76c86e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk9.ogg new file mode 100644 index 00000000..63b93248 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/dirt/dirt_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk1.ogg new file mode 100644 index 00000000..df88d31d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk10.ogg new file mode 100644 index 00000000..254c3fd0 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk2.ogg new file mode 100644 index 00000000..05d052d0 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk3.ogg new file mode 100644 index 00000000..c5256cde Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk4.ogg new file mode 100644 index 00000000..cb323156 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk5.ogg new file mode 100644 index 00000000..6e38b93d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk6.ogg new file mode 100644 index 00000000..e8aaaeea Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk7.ogg new file mode 100644 index 00000000..4efe1072 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk8.ogg new file mode 100644 index 00000000..8ae1b80a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk9.ogg new file mode 100644 index 00000000..3de487cf Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/grass/grass_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk1.ogg new file mode 100644 index 00000000..3127d136 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk10.ogg new file mode 100644 index 00000000..90fb8064 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk11.ogg new file mode 100644 index 00000000..d9ca9ac8 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk2.ogg new file mode 100644 index 00000000..08457ee1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk3.ogg new file mode 100644 index 00000000..11073b2f Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk4.ogg new file mode 100644 index 00000000..9cefaaf8 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk5.ogg new file mode 100644 index 00000000..be0458ca Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk6.ogg new file mode 100644 index 00000000..98f7e5e9 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk7.ogg new file mode 100644 index 00000000..df136948 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk8.ogg new file mode 100644 index 00000000..c93a3e8f Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk9.ogg new file mode 100644 index 00000000..49d2ea74 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/gravel/gravel_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk1.ogg new file mode 100644 index 00000000..eafd6289 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk10.ogg new file mode 100644 index 00000000..aa96a766 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk11.ogg new file mode 100644 index 00000000..7d7cd3b1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk2.ogg new file mode 100644 index 00000000..cc950f68 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk3.ogg new file mode 100644 index 00000000..56a79749 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk4.ogg new file mode 100644 index 00000000..5fac8649 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk5.ogg new file mode 100644 index 00000000..8b5479da Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk6.ogg new file mode 100644 index 00000000..16210c05 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk7.ogg new file mode 100644 index 00000000..10daa4b4 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk8.ogg new file mode 100644 index 00000000..76dc18be Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk9.ogg new file mode 100644 index 00000000..07bd5081 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbar/metalbar_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk1.ogg new file mode 100644 index 00000000..805a2713 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk2.ogg new file mode 100644 index 00000000..54fade91 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk3.ogg new file mode 100644 index 00000000..b85dde1c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk4.ogg new file mode 100644 index 00000000..ede7b7bb Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk5.ogg new file mode 100644 index 00000000..7550f38a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk6.ogg new file mode 100644 index 00000000..78bbb4c4 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk7.ogg new file mode 100644 index 00000000..7380c612 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk8.ogg new file mode 100644 index 00000000..86d6e29a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk9.ogg new file mode 100644 index 00000000..9e1d5168 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/metalbox/metalbox_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk1.ogg new file mode 100644 index 00000000..f0b6b435 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk2.ogg new file mode 100644 index 00000000..cb229013 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk3.ogg new file mode 100644 index 00000000..38beb82a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk4.ogg new file mode 100644 index 00000000..dda69300 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk5.ogg new file mode 100644 index 00000000..64919441 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk6.ogg new file mode 100644 index 00000000..e880811d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/mud/mud_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk1.ogg new file mode 100644 index 00000000..c2935a78 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk10.ogg new file mode 100644 index 00000000..fe403450 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk2.ogg new file mode 100644 index 00000000..872d22e6 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk3.ogg new file mode 100644 index 00000000..13e671a4 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk4.ogg new file mode 100644 index 00000000..be464b51 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk5.ogg new file mode 100644 index 00000000..fbceda9d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk6.ogg new file mode 100644 index 00000000..c8a75d4c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk7.ogg new file mode 100644 index 00000000..e793350c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk8.ogg new file mode 100644 index 00000000..a218fe02 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk9.ogg new file mode 100644 index 00000000..b1a01ca1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/muffledice/muffledice_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk1.ogg new file mode 100644 index 00000000..5f18763b Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk2.ogg new file mode 100644 index 00000000..8d8859ec Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk3.ogg new file mode 100644 index 00000000..6a767089 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk4.ogg new file mode 100644 index 00000000..6f01414a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/organic/organic_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk1.ogg new file mode 100644 index 00000000..1917b745 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk10.ogg new file mode 100644 index 00000000..a798803f Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk11.ogg new file mode 100644 index 00000000..67b27643 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk2.ogg new file mode 100644 index 00000000..a05e1a21 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk3.ogg new file mode 100644 index 00000000..cdaf75d9 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk4.ogg new file mode 100644 index 00000000..50ef2f16 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk5.ogg new file mode 100644 index 00000000..d7721add Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk6.ogg new file mode 100644 index 00000000..ed37c00a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk7.ogg new file mode 100644 index 00000000..c68d35cf Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk8.ogg new file mode 100644 index 00000000..9941188e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk9.ogg new file mode 100644 index 00000000..41c754db Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/rug/rug_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk1.ogg new file mode 100644 index 00000000..d0ff5527 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk10.ogg new file mode 100644 index 00000000..1e7112c4 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk11.ogg new file mode 100644 index 00000000..564aa4fb Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk2.ogg new file mode 100644 index 00000000..e4ba2c6f Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk3.ogg new file mode 100644 index 00000000..5e8b52cd Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk4.ogg new file mode 100644 index 00000000..9e8f5527 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk5.ogg new file mode 100644 index 00000000..1b2869c3 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk6.ogg new file mode 100644 index 00000000..cc509203 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk7.ogg new file mode 100644 index 00000000..7c7b5312 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk8.ogg new file mode 100644 index 00000000..e6184ae8 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk9.ogg new file mode 100644 index 00000000..8e07fd86 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/sand/sand_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk1.ogg new file mode 100644 index 00000000..3f64666c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk10.ogg new file mode 100644 index 00000000..81ac3882 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk11.ogg new file mode 100644 index 00000000..ccffdbdd Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk2.ogg new file mode 100644 index 00000000..d50bcb3c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk3.ogg new file mode 100644 index 00000000..519b6b05 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk4.ogg new file mode 100644 index 00000000..63ce49a1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk5.ogg new file mode 100644 index 00000000..0f6215ae Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk6.ogg new file mode 100644 index 00000000..c5eaac4f Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk7.ogg new file mode 100644 index 00000000..40a09bab Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk8.ogg new file mode 100644 index 00000000..5efd0a99 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk9.ogg new file mode 100644 index 00000000..2b8da0dd Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/snow/snow_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk1.ogg new file mode 100644 index 00000000..4fefacb2 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk10.ogg new file mode 100644 index 00000000..71bddeea Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk11.ogg new file mode 100644 index 00000000..725ddae1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk2.ogg new file mode 100644 index 00000000..38d41c9e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk3.ogg new file mode 100644 index 00000000..810fef06 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk4.ogg new file mode 100644 index 00000000..17c6fcbd Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk5.ogg new file mode 100644 index 00000000..884187e1 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk6.ogg new file mode 100644 index 00000000..a7d47e4e Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk7.ogg new file mode 100644 index 00000000..b5249645 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk8.ogg new file mode 100644 index 00000000..d7de99d5 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk9.ogg new file mode 100644 index 00000000..df29f98b Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/stone/stone_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk1.ogg new file mode 100644 index 00000000..886d656a Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk10.ogg new file mode 100644 index 00000000..8f5b2d75 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk11.ogg new file mode 100644 index 00000000..80cf9f63 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk2.ogg new file mode 100644 index 00000000..ddab7524 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk3.ogg new file mode 100644 index 00000000..342504a5 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk4.ogg new file mode 100644 index 00000000..058e98df Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk5.ogg new file mode 100644 index 00000000..f1a1ff79 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk6.ogg new file mode 100644 index 00000000..285874b0 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk7.ogg new file mode 100644 index 00000000..742ea879 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk8.ogg new file mode 100644 index 00000000..a5834a0d Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk9.ogg new file mode 100644 index 00000000..d76cdd7c Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/log_walk9.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk1.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk1.ogg new file mode 100644 index 00000000..51bfe179 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk1.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk10.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk10.ogg new file mode 100644 index 00000000..7b3c735b Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk10.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk11.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk11.ogg new file mode 100644 index 00000000..fa42038b Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk11.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk2.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk2.ogg new file mode 100644 index 00000000..f74bf6f6 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk2.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk3.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk3.ogg new file mode 100644 index 00000000..dd87c2f5 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk3.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk4.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk4.ogg new file mode 100644 index 00000000..8ac18bbd Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk4.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk5.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk5.ogg new file mode 100644 index 00000000..29460c10 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk5.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk6.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk6.ogg new file mode 100644 index 00000000..d2291783 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk6.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk7.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk7.ogg new file mode 100644 index 00000000..eedf3628 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk7.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk8.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk8.ogg new file mode 100644 index 00000000..a43f9dd3 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk8.ogg differ diff --git a/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk9.ogg b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk9.ogg new file mode 100644 index 00000000..4fc7cb58 Binary files /dev/null and b/common/src/main/resources/assets/dsurround/sounds/footsteps/wood/wood_walk9.ogg differ diff --git a/common/src/main/resources/dsurround.mixins.json b/common/src/main/resources/dsurround.mixins.json index 5c27f6bf..0842a487 100644 --- a/common/src/main/resources/dsurround.mixins.json +++ b/common/src/main/resources/dsurround.mixins.json @@ -27,7 +27,6 @@ "core.MixinEntity", "core.MixinEntityArrow", "core.MixinFogRenderer", - "core.MixinIngameHud", "core.MixinLivingEntity", "core.MixinParticleManager", "core.MixinRainSplashParticle", diff --git a/fabric/build.gradle b/fabric/build.gradle index c95ab1bd..6fcc0b5e 100644 --- a/fabric/build.gradle +++ b/fabric/build.gradle @@ -72,6 +72,10 @@ processResources { filesMatching('fabric.mod.json') { expand expandProps } + + into('/') { + from("$projectDir/../CREDITS.md") + } } shadowJar { diff --git a/fabric/src/main/java/org/orecruncher/fabric/mixins/MixinGui.java b/fabric/src/main/java/org/orecruncher/fabric/mixins/MixinGui.java new file mode 100644 index 00000000..a3e89844 --- /dev/null +++ b/fabric/src/main/java/org/orecruncher/fabric/mixins/MixinGui.java @@ -0,0 +1,33 @@ +package org.orecruncher.fabric.mixins; + +import net.minecraft.client.Minecraft; +import net.minecraft.client.gui.Gui; +import net.minecraft.client.gui.LayeredDraw; +import org.orecruncher.dsurround.gui.overlay.OverlayManager; +import org.orecruncher.dsurround.lib.di.ContainerManager; +import org.spongepowered.asm.mixin.Final; +import org.spongepowered.asm.mixin.Mixin; +import org.spongepowered.asm.mixin.Shadow; +import org.spongepowered.asm.mixin.injection.At; +import org.spongepowered.asm.mixin.injection.Inject; +import org.spongepowered.asm.mixin.injection.callback.CallbackInfo; + +@Mixin(Gui.class) +public class MixinGui { + + @Final + @Shadow + private LayeredDraw layers; + + @Final + @Shadow + private Minecraft minecraft; + + @Inject(method = "(Lnet/minecraft/client/Minecraft;)V", at = @At("RETURN")) + public void dsurround_constructor(Minecraft minecraftClient, CallbackInfo ci) { + // Add the overlay manager to the render layers of Gui + OverlayManager dsurround_overlayManager = ContainerManager.resolve(OverlayManager.class); + LayeredDraw layeredDraw = (new LayeredDraw()).add(dsurround_overlayManager::render); + this.layers.add(layeredDraw, () -> !this.minecraft.options.hideGui); + } +} diff --git a/fabric/src/main/resources/dsurround.mixins.fabric.json b/fabric/src/main/resources/dsurround.mixins.fabric.json new file mode 100644 index 00000000..a61f1c57 --- /dev/null +++ b/fabric/src/main/resources/dsurround.mixins.fabric.json @@ -0,0 +1,15 @@ +{ + "required": true, + "minVersion": "0.8", + "package": "org.orecruncher.fabric.mixins", + "compatibilityLevel": "JAVA_21", + "mixins": [ + ], + "client": [ + "MixinGui" + ], + "server": [], + "injectors": { + "defaultRequire": 1 + } +} diff --git a/fabric/src/main/resources/fabric.mod.json b/fabric/src/main/resources/fabric.mod.json index d8671640..7c2d0530 100644 --- a/fabric/src/main/resources/fabric.mod.json +++ b/fabric/src/main/resources/fabric.mod.json @@ -23,7 +23,8 @@ ] }, "mixins": [ - "dsurround.mixins.json" + "dsurround.mixins.json", + "dsurround.mixins.fabric.json" ], "depends": { "fabricloader": ">=0.16.9", diff --git a/gradle.properties b/gradle.properties index 14eb065e..975b7dd2 100644 --- a/gradle.properties +++ b/gradle.properties @@ -4,8 +4,8 @@ org.gradle.parallel=true # Mod properties mod_name = Dynamic Surroundings -mod_description = Alters the fabric of Minecraft experience by weaving a tapestry of sound and visual effects -mod_version = 0.4.1 +mod_description = Alters the fabric of Minecraft experience by weaving a tapestry of sound and visual effects +mod_version = 0.4.2 mod_author = OreCruncher mod_license = MIT mod_issues_url = https://github.com/OreCruncher/DynamicSurroundingsFabric/issues diff --git a/neoforge/build.gradle b/neoforge/build.gradle index 1cf523ac..5017cd21 100644 --- a/neoforge/build.gradle +++ b/neoforge/build.gradle @@ -71,6 +71,10 @@ processResources { filesMatching('META-INF/neoforge.mods.toml') { expand expandProps } + + into('/') { + from("$projectDir/../CREDITS.md") + } } shadowJar { diff --git a/neoforge/src/main/java/org/orecruncher/neoforge/NeoForgeMod.java b/neoforge/src/main/java/org/orecruncher/neoforge/NeoForgeMod.java index d3a1a7cd..80b50426 100644 --- a/neoforge/src/main/java/org/orecruncher/neoforge/NeoForgeMod.java +++ b/neoforge/src/main/java/org/orecruncher/neoforge/NeoForgeMod.java @@ -1,5 +1,6 @@ package org.orecruncher.neoforge; +import net.minecraft.resources.ResourceLocation; import net.neoforged.api.distmarker.Dist; import net.neoforged.bus.api.IEventBus; import net.neoforged.bus.api.SubscribeEvent; @@ -8,9 +9,12 @@ import net.neoforged.fml.common.Mod; import net.neoforged.fml.event.lifecycle.FMLClientSetupEvent; +import net.neoforged.neoforge.client.event.RegisterGuiLayersEvent; import net.neoforged.neoforge.client.gui.IConfigScreenFactory; import org.orecruncher.dsurround.Client; import org.orecruncher.dsurround.Constants; +import org.orecruncher.dsurround.gui.overlay.OverlayManager; +import org.orecruncher.dsurround.lib.di.ContainerManager; @Mod(value = Constants.MOD_ID, dist = Dist.CLIENT) public final class NeoForgeMod { @@ -19,6 +23,7 @@ public final class NeoForgeMod { public NeoForgeMod(ModContainer container, IEventBus modBus) { modBus.addListener(this::onInitializeClient); + modBus.addListener(this::onRegisterGuiLayersEvent); this.client = new Client(); this.client.construct(); @@ -26,6 +31,14 @@ public NeoForgeMod(ModContainer container, IEventBus modBus) { if (ModList.get().isLoaded(Constants.CLOTH_CONFIG_NEOFORGE)) container.registerExtensionPoint(IConfigScreenFactory.class, new ModConfigMenu()); + + } + + @SubscribeEvent + public void onRegisterGuiLayersEvent(RegisterGuiLayersEvent event) { + // Add the overlay manager to the render layers of Gui + OverlayManager dsurround_overlayManager = ContainerManager.resolve(OverlayManager.class); + event.registerBelowAll(ResourceLocation.fromNamespaceAndPath(Constants.MOD_ID, "layer/overlaymanager"), dsurround_overlayManager::render); } @SubscribeEvent diff --git a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/biomes.json b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/biomes.json index 66635789..fc287998 100644 --- a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/biomes.json +++ b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/biomes.json @@ -4,23 +4,75 @@ "_comment": "Music for Seasons", "acoustics": [ { - "factory": "dsurround_seasons:music/season/spring", + "factory": "dsurround_seasons:music/season/spring/early", "type": "music", + "weight": 5, "conditions": "season.isSpring()" }, { - "factory": "dsurround_seasons:music/season/summer", + "factory": "dsurround_seasons:music/season/spring/mid", "type": "music", + "weight": 5, + "conditions": "season.isSpring()" + }, + { + "factory": "dsurround_seasons:music/season/spring/late", + "type": "music", + "weight": 5, + "conditions": "season.isSpring()" + }, + { + "factory": "dsurround_seasons:music/season/summer/early", + "type": "music", + "weight": 5, + "conditions": "season.isSummer()" + }, + { + "factory": "dsurround_seasons:music/season/summer/mid", + "type": "music", + "weight": 5, + "conditions": "season.isSummer()" + }, + { + "factory": "dsurround_seasons:music/season/summer/late", + "type": "music", + "weight": 5, "conditions": "season.isSummer()" }, { - "factory": "dsurround_seasons:music/season/autumn", + "factory": "dsurround_seasons:music/season/autumn/early", "type": "music", + "weight": 5, "conditions": "season.isAutumn()" }, { - "factory": "dsurround_seasons:music/season/winter", + "factory": "dsurround_seasons:music/season/autumn/mid", + "type": "music", + "weight": 5, + "conditions": "season.isAutumn()" + }, + { + "factory": "dsurround_seasons:music/season/autumn/late", + "type": "music", + "weight": 5, + "conditions": "season.isAutumn()" + }, + { + "factory": "dsurround_seasons:music/season/winter/early", + "type": "music", + "weight": 5, + "conditions": "season.isWinter()" + }, + { + "factory": "dsurround_seasons:music/season/winter/mid", + "type": "music", + "weight": 5, + "conditions": "season.isWinter()" + }, + { + "factory": "dsurround_seasons:music/season/winter/late", "type": "music", + "weight": 5, "conditions": "season.isWinter()" } ] diff --git a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/sound_factories.json b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/sound_factories.json index de5df63d..dd89328b 100644 --- a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/sound_factories.json +++ b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/dsconfigs/sound_factories.json @@ -1,7 +1,7 @@ [ { - "location": "dsurround_seasons:music/season/spring", - "soundEvent": "dsurround_seasons:music.season.spring", + "location": "dsurround_seasons:music/season/spring/early", + "soundEvent": "dsurround_seasons:music.season.spring.early", "category": "MUSIC", "music": { "min_delay": 3000, @@ -10,8 +10,8 @@ } }, { - "location": "dsurround_seasons:music/season/summer", - "soundEvent": "dsurround_seasons:music.season.summer", + "location": "dsurround_seasons:music/season/spring/mid", + "soundEvent": "dsurround_seasons:music.season.spring.mid", "category": "MUSIC", "music": { "min_delay": 3000, @@ -20,8 +20,8 @@ } }, { - "location": "dsurround_seasons:music/season/autumn", - "soundEvent": "dsurround_seasons:music.season.autumn", + "location": "dsurround_seasons:music/season/spring/late", + "soundEvent": "dsurround_seasons:music.season.spring.late", "category": "MUSIC", "music": { "min_delay": 3000, @@ -30,8 +30,86 @@ } }, { - "location": "dsurround_seasons:music/season/winter", - "soundEvent": "dsurround_seasons:music.season.winter", + "location": "dsurround_seasons:music/season/summer/early", + "soundEvent": "dsurround_seasons:music.season.summer.early", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/summer/mid", + "soundEvent": "dsurround_seasons:music.season.summer.mid", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/summer/late", + "soundEvent": "dsurround_seasons:music.season.summer.late", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/autumn/early", + "soundEvent": "dsurround_seasons:music.season.autumn.early", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/autumn/mid", + "soundEvent": "dsurround_seasons:music.season.autumn.mid", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/autumn/late", + "soundEvent": "dsurround_seasons:music.season.autumn.late", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + }, + { + "location": "dsurround_seasons:music/season/winter/early", + "soundEvent": "dsurround_seasons:music.season.winter.early", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + { + "location": "dsurround_seasons:music/season/winter/mid", + "soundEvent": "dsurround_seasons:music.season.winter.mid", + "category": "MUSIC", + "music": { + "min_delay": 3000, + "max_delay": 12000, + "replace_current_music": false + } + { + "location": "dsurround_seasons:music/season/winter/late", + "soundEvent": "dsurround_seasons:music.season.winter.late", "category": "MUSIC", "music": { "min_delay": 3000, diff --git a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/sounds.json b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/sounds.json index 889eaac7..477ccedc 100644 --- a/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/sounds.json +++ b/packs/Dynamic Surroundings - Seasons/assets/dsurround_seasons/sounds.json @@ -1,11 +1,11 @@ { - "music.season.spring": { - "ds_title": "The Four Seasons - Spring", + "music.season.spring.early": { + "ds_title": "The Four Seasons - Early Spring", "ds_category": "music", "ds_credits": [ { - "name": "The Four Seasons Concerto No. 1 in E Major", - "author": "Vivaldi, arranged by Jerzey Kulik", + "name": "Concerto No. 1 in E major, Op. 8, RV 269, \"Spring\" (La primavera), Allegro (in E major)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", "license": "Open Source Audio" } @@ -15,12 +15,40 @@ "name": "dsurround_seasons:music/spring1", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.spring.mid": { + "ds_title": "The Four Seasons - Mid Spring", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 1 in E major, Op. 8, RV 269, \"Spring\" (La primavera), Largo e pianissimo sempre (in C♯ minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/spring2", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.spring.late": { + "ds_title": "The Four Seasons - Late Spring", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 1 in E major, Op. 8, RV 269, \"Spring\" (La primavera), Allegro pastorale (in E major)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/spring3", "stream": true, @@ -28,13 +56,13 @@ } ] }, - "music.season.summer": { - "ds_title": "The Four Seasons - Summer", + "music.season.summer.early": { + "ds_title": "The Four Seasons - Early Summer", "ds_category": "music", "ds_credits": [ { - "name": "The Four Seasons Concerto No. 2 in G Minor", - "author": "Vivaldi, arranged by Jerzey Kulik", + "name": "Concerto No. 2 in G minor, Op. 8, RV 315, \"Summer\" (L'estate), Allegro non molto (in G minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", "license": "Open Source Audio" } @@ -44,12 +72,40 @@ "name": "dsurround_seasons:music/summer1", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.summer.mid": { + "ds_title": "The Four Seasons - Mid Summer", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 2 in G minor, Op. 8, RV 315, \"Summer\" (L'estate), Adagio e piano – Presto e forte (in G minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/summer2", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.summer.late": { + "ds_title": "The Four Seasons - Late Summer", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 2 in G minor, Op. 8, RV 315, \"Summer\" (L'estate), Presto (in G minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/summer3", "stream": true, @@ -57,13 +113,13 @@ } ] }, - "music.season.autumn": { - "ds_title": "The Four Seasons - Autumn", + "music.season.autumn.early": { + "ds_title": "The Four Seasons - Early Autumn", "ds_category": "music", "ds_credits": [ { - "name": "The Four Seasons Concerto No. 3 in F Major", - "author": "Vivaldi, arranged by Jerzey Kulik", + "name": "Concerto No. 3 in F major, Op. 8, RV 293, \"Autumn\" (L'autunno), Allegro (in F major)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", "license": "Open Source Audio" } @@ -73,12 +129,40 @@ "name": "dsurround_seasons:music/autumn1", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.autumn.mid": { + "ds_title": "The Four Seasons - Mid Autumn", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 3 in F major, Op. 8, RV 293, \"Autumn\" (L'autunno), Adagio molto (in D minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/autumn2", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.autumn.late": { + "ds_title": "The Four Seasons - Late Autumn", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 3 in F major, Op. 8, RV 293, \"Autumn\" (L'autunno), Allegro (in F major)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/autumn3", "stream": true, @@ -86,13 +170,13 @@ } ] }, - "music.season.winter": { - "ds_title": "The Four Seasons - Winter", + "music.season.winter.early": { + "ds_title": "The Four Seasons - Early Winter", "ds_category": "music", "ds_credits": [ { - "name": "The Four Seasons Concerto No. 4 in F Minor", - "author": "Vivaldi, arranged by Jerzey Kulik", + "name": "Concerto No. 4 in F minor, Op. 8, RV 297, \"Winter\" (L'inverno), Allegro non molto (in F minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", "license": "Open Source Audio" } @@ -102,12 +186,40 @@ "name": "dsurround_seasons:music/winter1", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.winter.mid": { + "ds_title": "The Four Seasons - Mid Winter", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 4 in F minor, Op. 8, RV 297, \"Winter\" (L'inverno), Largo (in E♭ major)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/winter2", "stream": true, "volume": 0.5 - }, + } + ] + }, + "music.season.winter.late": { + "ds_title": "The Four Seasons - Late Winter", + "ds_category": "music", + "ds_credits": [ + { + "name": "Concerto No. 4 in F minor, Op. 8, RV 297, \"Winter\" (L'inverno), Allegro (in F minor)", + "author": "Antonio Vivaldi, arranged by Jerzey Kulik", + "website": "https://archive.org/details/ETitleA.Vivaldi-TheFourSeasons", + "license": "Open Source Audio" + } + ], + "sounds": [ { "name": "dsurround_seasons:music/winter3", "stream": true, diff --git a/versions.json b/versions.json index 7b299b8c..0c82b3c8 100644 --- a/versions.json +++ b/versions.json @@ -2,6 +2,7 @@ "downloadLocation": "https://www.curseforge.com/minecraft/mc-mods/dynamic-surroundings/files", "releases": { "1.21.1": { + "0.4.2": "https://github.com/OreCruncher/DynamicSurroundingsFabric/blob/main/CHANGELOG.md#dynamicsurroundings-1211-042", "0.4.1": "https://github.com/OreCruncher/DynamicSurroundingsFabric/blob/main/CHANGELOG.md#dynamicsurroundings-1211-041", "0.4.0": "https://github.com/OreCruncher/DynamicSurroundingsFabric/blob/main/CHANGELOG.md#dynamicsurroundings-1211-040" }, @@ -34,7 +35,7 @@ } }, "recommend": { - "1.21.1": "0.4.0", + "1.21.1": "0.4.2", "1.20.4": "0.3.3", "1.20.1": "0.3.3", "1.17.1": "0.0.4"