This commit is contained in:
Minecon724 2024-12-18 19:59:07 +01:00
parent ff747c9013
commit 29ed0b6a13
Signed by: Minecon724
GPG key ID: 3CCC4D267742C8E8
10 changed files with 203 additions and 153 deletions

View file

@ -14,6 +14,10 @@ public class NotEncoder implements OpusEncoder {
this.size = opusFrames.length; this.size = opusFrames.length;
} }
public void seek(int frame) {
i = Math.clamp(frame, 0, size);
}
public boolean hasRemaining() { public boolean hasRemaining() {
return i < size; return i < size;
} }

View file

@ -40,77 +40,53 @@ public class ItemEvents implements Listener {
} }
} else { } else {
if (right) { if (right) {
System.out.println("shirt + rmb open inventory"); System.out.println("shift + rmb bind spekar");
// TODO open inventory
} else {
System.out.println("shift + lmb bind spekar");
var block = event.getClickedBlock(); var block = event.getClickedBlock();
if (block != null) { if (block != null) {
if (block.getType() == Material.NOTE_BLOCK) { if (block.getType() == Material.NOTE_BLOCK) {
var speaker = BlockSpeaker.create(block.getLocation()); var location = block.getLocation().add(0.5, 0.5, 0.5);
if (speaker != null) { var speaker = BlockSpeaker.create(location);
player.setSpeaker(speaker);
block.getWorld().spawnParticle(Particle.HAPPY_VILLAGER, block.getLocation(), 10); player.setSpeaker(speaker);
event.setCancelled(true); // prevent break block.getWorld().spawnParticle(Particle.HAPPY_VILLAGER, block.getLocation(), 10);
} else {
event.getPlayer().sendMessage("This speaker is occupied"); event.setCancelled(true); // prevent break
}
} }
} }
} }
// originally inventory was supposed to be here
} }
} }
@EventHandler
public void onDeath(EntityDeathEvent event) {
if (event.getEntity() instanceof Item item) {
var id = PortableMediaPlayer.idFromItemStack(item.getItemStack());
if (id == -1) return;
System.out.println("A player ded");
Speakers.destroy(id);
}
}
@EventHandler
public void onDespawn(ItemDespawnEvent event) {
var id = PortableMediaPlayer.idFromItemStack(event.getEntity().getItemStack());
if (id == -1) return;
System.out.println("A player despeawnd");
Speakers.destroy(id);
}
@EventHandler @EventHandler
public void onDrop(PlayerDropItemEvent event) { public void onDrop(PlayerDropItemEvent event) {
var id = PortableMediaPlayer.idFromItemStack(event.getItemDrop().getItemStack()); var item = PortableMediaPlayer.fromItemStack(event.getItemDrop().getItemStack());
if (id == -1) return; if (item == null) return;
System.out.println("A playe rdropped"); System.out.println("A playe rdropped");
Speakers.attachTo(id, event.getItemDrop()); item.stop();
} }
@EventHandler @EventHandler
public void onPickup(EntityPickupItemEvent event) { public void onPickup(EntityPickupItemEvent event) {
var id = PortableMediaPlayer.idFromItemStack(event.getItem().getItemStack()); var item = PortableMediaPlayer.fromItemStack(event.getItem().getItemStack());
if (id == -1) return; if (item == null) return;
System.out.println("A player puckuperd"); System.out.println("A player puckuperd");
Speakers.attachTo(id, event.getEntity()); // TODO do somehting?
} }
public void onMove(InventoryMoveItemEvent event) { public void onMove(InventoryMoveItemEvent event) {
if (event.getDestination().getType() == InventoryType.PLAYER) { if (event.getDestination().getType() == InventoryType.PLAYER) {
var id = PortableMediaPlayer.idFromItemStack(event.getItem()); var item = PortableMediaPlayer.fromItemStack(event.getItem());
if (id == -1) return; if (item == null) return;
System.out.println("A player storaged :(("); System.out.println("A player storaged :((");
Speakers.destroy(id); item.stop();
} }
} }
} }

View file

@ -2,7 +2,6 @@ package eu.m724.musicPlugin.item;
import eu.m724.musicPlugin.item.speaker.Speaker; import eu.m724.musicPlugin.item.speaker.Speaker;
import eu.m724.musicPlugin.player.MovingMusicPlayer; import eu.m724.musicPlugin.player.MovingMusicPlayer;
import eu.m724.musicPlugin.player.MusicPlayer;
import net.md_5.bungee.api.ChatColor; import net.md_5.bungee.api.ChatColor;
import org.bukkit.Material; import org.bukkit.Material;
import org.bukkit.NamespacedKey; import org.bukkit.NamespacedKey;
@ -17,8 +16,8 @@ import java.util.List;
import java.util.concurrent.ThreadLocalRandom; import java.util.concurrent.ThreadLocalRandom;
public class PortableMediaPlayer { public class PortableMediaPlayer {
private static NamespacedKey idKey = new NamespacedKey("tweaks724", "player_id"); static NamespacedKey idKey = new NamespacedKey("tweaks724", "player_id");
private static NamespacedKey dataKey = new NamespacedKey("tweaks724", "player_data"); static NamespacedKey dataKey = new NamespacedKey("tweaks724", "player_data");
public final int id; public final int id;
@ -29,10 +28,7 @@ public class PortableMediaPlayer {
public final boolean premium; public final boolean premium;
public final String engraving; public final String engraving;
private final MovingMusicPlayer musicPlayer = new MovingMusicPlayer(); PortableMediaPlayer(int id, int storageSeconds, int audioBitrate, boolean premium, String engraving) {
private Speaker linkedSpeaker;
public PortableMediaPlayer(int id, int storageSeconds, int audioBitrate, boolean premium, String engraving) {
this.id = id; this.id = id;
this.storageSeconds = storageSeconds; this.storageSeconds = storageSeconds;
this.audioBitrate = audioBitrate; this.audioBitrate = audioBitrate;
@ -44,22 +40,33 @@ public class PortableMediaPlayer {
return new PortableMediaPlayer(ThreadLocalRandom.current().nextInt(), 600, 32000, premium, engraving); return new PortableMediaPlayer(ThreadLocalRandom.current().nextInt(), 600, 32000, premium, engraving);
} }
public void stop() {
Speakers.speakers.remove(id).destroy();
getMusicPlayer().stop();
Speakers.players.remove(id);
}
public void setSpeaker(Speaker speaker) { public void setSpeaker(Speaker speaker) {
Speakers.speakers.put(id, ) var linked = Speakers.speakers.get(id);
if (speaker.equals(linkedSpeaker)) if (speaker.equals(linked))
return; return;
if (linkedSpeaker != null) if (linked != null)
linkedSpeaker.destroy(); linked.destroy();
linkedSpeaker = speaker; var paused = getMusicPlayer().isPlaying();
speaker.setMusicPlayer(musicPlayer); speaker.setMusicPlayer(getMusicPlayer());
speaker.setDestroyCallback(c -> this.onSpeakerDestroyed());
if (paused) getMusicPlayer().unpause();
Speakers.speakers.put(id, speaker);
} }
public MusicPlayer getMusicPlayer() { public MovingMusicPlayer getMusicPlayer() {
return musicPlayer; return Speakers.players.computeIfAbsent(id, m -> new MovingMusicPlayer());
} }
/* Item functions */
public ItemStack getItemStack() { public ItemStack getItemStack() {
var is = new ItemStack(Material.IRON_INGOT); var is = new ItemStack(Material.IRON_INGOT);
var meta = is.getItemMeta(); var meta = is.getItemMeta();
@ -77,29 +84,6 @@ public class PortableMediaPlayer {
return is; return is;
} }
public static int idFromItemStack(ItemStack itemStack) {
var meta = itemStack.getItemMeta();
if (meta == null) return -1;
var id = meta.getPersistentDataContainer().get(idKey, PersistentDataType.INTEGER);
if (id == null) return -1;
return id;
}
public static PortableMediaPlayer fromItemStack(ItemStack itemStack) {
var meta = itemStack.getItemMeta();
if (meta == null) return null;
var id = meta.getPersistentDataContainer().get(idKey, PersistentDataType.INTEGER);
if (id == null) return null;
var data = meta.getPersistentDataContainer().get(dataKey, PersistentDataType.BYTE_ARRAY);
if (data == null) return null;
return fromData(id, data);
}
private byte[] getData() { private byte[] getData() {
var buffer = ByteBuffer.allocate(11 + engraving.length()); var buffer = ByteBuffer.allocate(11 + engraving.length());
buffer.put((byte) 0); // version buffer.put((byte) 0); // version
@ -113,23 +97,4 @@ public class PortableMediaPlayer {
return buffer.array(); return buffer.array();
} }
private static PortableMediaPlayer fromData(int id, byte[] data) {
ByteBuffer buffer = ByteBuffer.wrap(data);
if (buffer.get() != 0) { // TODO version mismatch
}
var premium = buffer.get() == 1;
var storageSeconds = buffer.getShort() & 0xFFFF;
var audioBitrate = (buffer.get() & 0xFF) * 1000;
var eb = new byte[buffer.get() & 0xFF];
buffer.get(eb);
var engraving = new String(eb, StandardCharsets.UTF_8);
return new PortableMediaPlayer(id, storageSeconds, audioBitrate, premium, engraving);
}
} }

View file

@ -0,0 +1,60 @@
package eu.m724.musicPlugin.item;
import org.bukkit.inventory.ItemStack;
import org.bukkit.persistence.PersistentDataType;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.util.HashMap;
import java.util.Map;
// This class stores PortableMediaPlayer instances so that there's no hassle of working with a new one every time
public class PortableMediaPlayers {
private static Map<Integer, PortableMediaPlayer> players = new HashMap<>();
public static PortableMediaPlayer get(Integer id) {
return players.get(id);
}
public static PortableMediaPlayer get(ItemStack itemStack) {
var meta = itemStack.getItemMeta();
if (meta == null) return null;
var id = meta.getPersistentDataContainer().get(PortableMediaPlayer.idKey, PersistentDataType.INTEGER);
if (id == null) return null;
// check cache
var player = players.get(id);
if (player != null) return player;
// if not cached
var data = meta.getPersistentDataContainer().get(PortableMediaPlayer.dataKey, PersistentDataType.BYTE_ARRAY);
if (data == null) return null;
player = fromData(id, data);
players.put(id, player);
return player;
}
private static PortableMediaPlayer fromData(int id, byte[] data) {
ByteBuffer buffer = ByteBuffer.wrap(data);
if (buffer.get() != 0) { // TODO version mismatch
}
var premium = buffer.get() == 1;
var storageSeconds = buffer.getShort() & 0xFFFF;
var audioBitrate = (buffer.get() & 0xFF) * 1000;
var eb = new byte[buffer.get() & 0xFF];
buffer.get(eb);
var engraving = new String(eb, StandardCharsets.UTF_8);
return new PortableMediaPlayer(id, storageSeconds, audioBitrate, premium, engraving);
}
}

View file

@ -1,11 +1,7 @@
package eu.m724.musicPlugin.item; package eu.m724.musicPlugin.item;
import eu.m724.musicPlugin.item.speaker.BlockSpeaker;
import eu.m724.musicPlugin.item.speaker.Speaker; import eu.m724.musicPlugin.item.speaker.Speaker;
import eu.m724.musicPlugin.player.MovingMusicPlayer; import eu.m724.musicPlugin.player.MovingMusicPlayer;
import org.bukkit.Location;
import org.bukkit.block.Block;
import org.bukkit.entity.Entity;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
@ -13,21 +9,4 @@ import java.util.Map;
public class Speakers { public class Speakers {
static Map<Integer, Speaker> speakers = new HashMap<>(); static Map<Integer, Speaker> speakers = new HashMap<>();
static Map<Integer, MovingMusicPlayer> players = new HashMap<>(); static Map<Integer, MovingMusicPlayer> players = new HashMap<>();
static Speaker get(int id) {
return speakers.get(id);
}
static void destroy(int id) {
var s = speakers.remove(id);
if (s != null) s.destroy();
}
static void moveTo(int id, Location location) {
//speakers.computeIfAbsent(id, i -> new Speaker()).moveTo(location);
}
static void attachTo(int id, Entity entity) {
//speakers.computeIfAbsent(id, i -> new Speaker()).attachTo(entity);
}
} }

View file

@ -1,8 +1,6 @@
package eu.m724.musicPlugin.item.speaker; package eu.m724.musicPlugin.item.speaker;
import eu.m724.musicPlugin.player.MovingMusicPlayer; import eu.m724.musicPlugin.player.MovingMusicPlayer;
import eu.m724.musicPlugin.player.MusicPlayer;
import eu.m724.musicPlugin.player.StaticMusicPlayer;
import org.bukkit.Location; import org.bukkit.Location;
import java.util.HashMap; import java.util.HashMap;

View file

@ -2,24 +2,37 @@ package eu.m724.musicPlugin.item.speaker;
import eu.m724.musicPlugin.player.MovingMusicPlayer; import eu.m724.musicPlugin.player.MovingMusicPlayer;
import java.util.function.Consumer;
public abstract class Speaker { public abstract class Speaker {
private MovingMusicPlayer musicPlayer; private MovingMusicPlayer musicPlayer;
private Consumer<Void> destroyCallback;
public MovingMusicPlayer getMusicPlayer() { public MovingMusicPlayer getMusicPlayer() {
return musicPlayer; return musicPlayer;
} }
public void setMusicPlayer(MovingMusicPlayer musicPlayer) { public void setMusicPlayer(MovingMusicPlayer musicPlayer) {
if (musicPlayer != null) { if (musicPlayer.equals(this.musicPlayer)) return;
if (this.musicPlayer != null) {
musicPlayer.pause(); musicPlayer.pause();
} }
this.musicPlayer = musicPlayer; this.musicPlayer = musicPlayer;
onSetMusicPlayer(musicPlayer);
} }
public void destroy() { public void destroy() {
this.musicPlayer.stop(); this.musicPlayer.pause();
this.musicPlayer = null; this.musicPlayer = null;
onDestroy();
destroyCallback.accept(null);
}
public void setDestroyCallback(Consumer<Void> consumer) {
this.destroyCallback = consumer;
} }
abstract void onSetMusicPlayer(MovingMusicPlayer musicPlayer); abstract void onSetMusicPlayer(MovingMusicPlayer musicPlayer);

View file

@ -0,0 +1,38 @@
package eu.m724.musicPlugin.player;
import de.maxhenkel.voicechat.api.ServerLevel;
import de.maxhenkel.voicechat.api.VoicechatConnection;
import de.maxhenkel.voicechat.api.audiochannel.AudioChannel;
import eu.m724.musicPlugin.Statics;
import org.bukkit.entity.Player;
import java.util.UUID;
public class LocalMusicPlayer extends MusicPlayer {
private final ServerLevel level;
private final VoicechatConnection connection;
public LocalMusicPlayer(Player player) throws NotConnectedException {
this.level = Statics.api.fromServerLevel(player.getWorld());
this.connection = Statics.api.getConnectionOf(player.getUniqueId());
if (connection == null) {
throw new NotConnectedException();
}
}
@Override
AudioChannel createChannel() {
var channel = api.createStaticAudioChannel(
UUID.randomUUID(),
level,
connection
);
channel.setCategory("musicc");
return channel;
}
public static class NotConnectedException extends Exception {}
}

View file

@ -1,7 +1,6 @@
package eu.m724.musicPlugin.player; package eu.m724.musicPlugin.player;
import de.maxhenkel.voicechat.api.audiochannel.AudioChannel; import de.maxhenkel.voicechat.api.audiochannel.AudioChannel;
import de.maxhenkel.voicechat.api.audiochannel.LocationalAudioChannel;
import org.bukkit.Location; import org.bukkit.Location;
import org.bukkit.entity.Entity; import org.bukkit.entity.Entity;
@ -12,18 +11,14 @@ public class MovingMusicPlayer extends MusicPlayer {
private Location location; private Location location;
public void moveTo(Location location) { public void moveTo(Location location) {
if (channel == null || !location.getWorld().equals(this.location.getWorld())) { System.out.println("Mvoed to a");
this.location = location;
this.channel = createChannel();
if (isPlaying()) { this.location = location;
this.pause(); this.channel = createChannel();
this.unpause();
} if (isReady() && !isPlaying()) {
} else { this.pause();
((LocationalAudioChannel)channel).updateLocation( this.unpause();
api.createPosition(location.getX(), location.getY(), location.getZ())
);
} }
} }

View file

@ -3,6 +3,7 @@ package eu.m724.musicPlugin.player;
import de.maxhenkel.voicechat.api.VoicechatServerApi; import de.maxhenkel.voicechat.api.VoicechatServerApi;
import de.maxhenkel.voicechat.api.audiochannel.AudioChannel; import de.maxhenkel.voicechat.api.audiochannel.AudioChannel;
import de.maxhenkel.voicechat.api.audiochannel.AudioPlayer; import de.maxhenkel.voicechat.api.audiochannel.AudioPlayer;
import de.maxhenkel.voicechat.api.audiochannel.LocationalAudioChannel;
import eu.m724.musicPlugin.Statics; import eu.m724.musicPlugin.Statics;
import eu.m724.musicPlugin.file.AudioFile; import eu.m724.musicPlugin.file.AudioFile;
@ -15,7 +16,7 @@ public abstract class MusicPlayer {
AudioChannel channel; AudioChannel channel;
private boolean playing = false, paused = false; private boolean ready = false, playing = false;
private AudioPlayer player; private AudioPlayer player;
private AudioFile audioFile; private AudioFile audioFile;
@ -54,7 +55,7 @@ public abstract class MusicPlayer {
if (audioFile != null) if (audioFile != null)
this.audioFile = audioFile; this.audioFile = audioFile;
if (playing) if (ready)
stop(); stop();
if (!audioFile.isLoaded()) { if (!audioFile.isLoaded()) {
@ -65,11 +66,15 @@ public abstract class MusicPlayer {
} }
} }
ready = true;
playing = true;
unpause(); unpause();
} }
public void unpause() { public void unpause() {
if (playing) return; if (!ready || !playing) return;
System.out.println("Unapised");
var sa = new short[960]; var sa = new short[960];
var enc = audioFile.getEncoder(); var enc = audioFile.getEncoder();
@ -79,18 +84,18 @@ public abstract class MusicPlayer {
player.startPlaying(); player.startPlaying();
// TODO this may take a long time and it might glitch if it does // TODO this may take a long time and it might glitch if it does
onAction.accept(playing ? TrackAction.UNPAUSE : TrackAction.START); onAction.accept(ready ? TrackAction.UNPAUSE : TrackAction.START);
playing = true; playing = false;
paused = false;
System.out.println("unpaused");
} }
/** /**
* Stops playback and rewinds * Stops playback and rewinds
*/ */
public void stop() { public void stop() {
if (!playing) return; if (!ready) return;
stopPlayback(false); stopPlayback(false);
audioFile.getEncoder().resetState(); audioFile.getEncoder().resetState();
@ -100,18 +105,27 @@ public abstract class MusicPlayer {
* Pauses playback * Pauses playback
*/ */
public void pause() { public void pause() {
if (!ready || playing) return;
stopPlayback(true); stopPlayback(true);
} }
/**
* Seek to some point in current track
*
* @param target the target time in milliseconds
*/
public void seek(int target) {
audioFile.getEncoder().seek(target / 20); // a frame is 20 ms usually
}
/* Internal methods */ /* Internal methods */
private void stopPlayback(boolean pause) { private void stopPlayback(boolean pause) {
if (!playing) return; System.out.println("psued");
playing = pause;
paused = pause; ready = pause;
playing = false;
player.stopPlaying(); player.stopPlaying();
player = null; player = null;
@ -120,13 +134,17 @@ public abstract class MusicPlayer {
} }
private void onStop() { private void onStop() {
if (paused) // paused if (playing) { // paused
System.out.println("I detected pause");
onAction.accept(TrackAction.PAUSE); onAction.accept(TrackAction.PAUSE);
else if (playing) { // not paused and still playing } else if (ready) { // not paused and still playing
System.out.println("I detected end");
onAction.accept(TrackAction.DURATION); onAction.accept(TrackAction.DURATION);
playing = false; ready = false;
} else // not playing } else { // not playing
System.out.println("I detected stop");
onAction.accept(TrackAction.STOP); onAction.accept(TrackAction.STOP);
}
} }
public enum TrackAction { public enum TrackAction {
@ -152,7 +170,11 @@ public abstract class MusicPlayer {
UNPAUSE UNPAUSE
} }
public boolean isReady() {
return ready;
}
public boolean isPlaying() { public boolean isPlaying() {
return playing; return ready && playing;
} }
} }