Jump to content

Recommended Posts

Posted (edited)

Hello,

I am trying to send a string name (which is stored in NBT data in my tile entity) to the client screen, but I don't know where/how I should call the send network package and how to get it in the screen class.

My Packet class:

Spoiler
public class PacketSendName {
    private final String name;

    public PacketSendName(PacketBuffer buf){
        this.name = buf.readString();
    }

    public PacketSendName(String name){
        this.name = name;
    }

    public void toBytes(PacketBuffer buf){
        buf.writeString(name);
    }

    public void handle(Supplier<NetworkEvent.Context> ctx){
        ctx.get().enqueueWork(() ->{
            World world = Objects.requireNonNull(ctx.get().getSender()).world;

        });
        
        //...

        ctx.get().setPacketHandled(true);
    }
}

 

I can access the name from the tile entity via tileEntity.getName()

In my screen class I just call this.container.getTileEntity().getName() to get the name.

Saving the name with nbt works fine. I just want to know how I can get the name from the tile entity

 

In my block class I called

Network.INSTANCE.send(new PacketSendName(tileEntity.getName()), Network.INSTANCE, NetworkDirection.PLAY_TO_CLIENT);

but it is an error because SimpleChannel is not compatible with NetworkManager. I don't know which variable I should pass instead. And I questioned as well: Would it make more sense to pass a position parameter, so it can get to the right player or screen?

 

Thanks for your help

 

EDIT: Maybe you can name an example where Mojangs is using it in that way ^^

Edited by J3ramy
Posted
1 hour ago, J3ramy said:

but it is an error because SimpleChannel is not compatible with NetworkManager.

which kind of error

1 hour ago, J3ramy said:

I don't know which variable I should pass instead.

what variables are you talking about and please show your Network class

1 hour ago, J3ramy said:

And I questioned as well: Would it make more sense to pass a position parameter, so it can get to the right player or screen?

on the client there is only one player which is able to have a Screen (LocalPlayer), you can get it from Minecraft.getInstance().player

Spoiler
1 hour ago, J3ramy said:
    public void handle(Supplier<NetworkEvent.Context> ctx){
        ctx.get().enqueueWork(() ->{
            World world = Objects.requireNonNull(ctx.get().getSender()).world;

        });
        
        //...

        ctx.get().setPacketHandled(true);
    }

first of all there is no sender on the client,
you also should handle client actions in a separate class to avoid conflicts with client only classes

Posted
Quote

which kind of error

The error is "Simple Channel is not compatible with NetworkManager", so it means that the second parameter is the wrong type but I don't which variable I should pass instead of Network.INSTANCE:

Network.INSTANCE.send(new PacketSendName(tileEntity.getName()), Network.INSTANCE, NetworkDirection.PLAY_TO_CLIENT);
Quote

what variables are you talking about and please show your Network class

Spoiler
public class Network {
    public static SimpleChannel INSTANCE;
    private static int ID;

    public static int getNextId(){
        return ID++;
    }

    public static void registerMessages(){
        INSTANCE = NetworkRegistry.newSimpleChannel(new ResourceLocation(CoasterideMod.MOD_ID, "network"), () -> "1.0", s -> true, s -> true);

        INSTANCE.registerMessage(getNextId(), PacketBopTargetSpeed.class, PacketBopTargetSpeed::toBytes, PacketBopTargetSpeed::new, PacketBopTargetSpeed::handle);
        INSTANCE.registerMessage(getNextId(), PacketBopChangePanelState.class, PacketBopChangePanelState::toBytes, PacketBopChangePanelState::new, PacketBopChangePanelState::handle);
        INSTANCE.registerMessage(getNextId(), PacketBopButtonInteraction.class, PacketBopButtonInteraction::toBytes, PacketBopButtonInteraction::new, PacketBopButtonInteraction::handle);

        INSTANCE.registerMessage(getNextId(), PacketLopLiftSpeed.class, PacketLopLiftSpeed::toBytes, PacketLopLiftSpeed::new, PacketLopLiftSpeed::handle);
        INSTANCE.registerMessage(getNextId(), PacketLopChangePanelState.class, PacketLopChangePanelState::toBytes, PacketLopChangePanelState::new, PacketLopChangePanelState::handle);
        INSTANCE.registerMessage(getNextId(), PacketLopButtonInteraction.class, PacketLopButtonInteraction::toBytes, PacketLopButtonInteraction::new, PacketLopButtonInteraction::handle);

        INSTANCE.registerMessage(getNextId(), PacketPlaySound.class, PacketPlaySound::toBytes, PacketPlaySound::new, PacketPlaySound::handle);
    }
}

 

 

Posted

you should use the SimpleChannel#send method with two parameters, use PacketDistributor.PLAYER (as first parameter)
and call #with and pass in a Suppiler with the player you want to send the Packet to

Posted

Nice thanks.

I changed the line to

Network.INSTANCE.send(PacketDistributor.PLAYER.with(() -> (ServerPlayerEntity) player), new PacketSendName(tileEntity.getName()));

and it calls the handler method.

But how do I now pass this to the screen container class?

Posted

you can get the current Screen via:

Minecraft.getInstance().screen

you then need to check via instanceof if the Screen is an instance of your Screen,
then you can set the data via setters and then you be able to use it in the your Screen for rendering etc.

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.
Note: Your post will require moderator approval before it will be visible.

Guest
Unfortunately, your content contains terms that we do not allow. Please edit your content to remove the highlighted words below.
Reply to this topic...

×   Pasted as rich text.   Restore formatting

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

Announcements



  • Recently Browsing

    • No registered users viewing this page.
  • Posts

    • I see. I'm sure I tried again with a lower case word, a few times. I ended up doing the same thing as pixxy in the end. Is this beyond the scope of the admins to fix? Are they already aware of it, or should I use 'contact us' to post a ticket?
    • I’m working on a Manta Ray entity in MCreator using GeckoLib animations, and my goal is to have a looping (flip) animation that ends at −360°, then transitions seamlessly into a swim animation starting at 0°. However, every method I’ve tried—like quickly interpolating the angle, inserting a brief keyframe at 0°, or using a micro “bridge” animation—still causes a visible “flash” https://imgur.com/a/5ucjUb9 or "jump" when the rotation resets. I want a perfectly smooth motion from the flip’s final rotation to the swim’s initial rotation. If anyone has solved this in MCreator/GeckoLib, or found a better trick for handling the −360° →0° gap without a snap, I’d appreciate some advice ! P.S.- I cannot set swim to start at -360 because I would have the same issue but in reverse. Here's the custom LoopingAnimationGoal :   class LoopingAnimationGoal extends Goal { private final MantaRayEntity entity; private final int cooldownTime; private int animationTimer; private int cooldownTimer; // New boolean to prevent double calls private boolean isLoopingActive = false; public LoopingAnimationGoal(MantaRayEntity entity, int cooldownTime) { this.entity = entity; this.cooldownTime = cooldownTime; this.animationTimer = 0; this.cooldownTimer = 0; this.setFlags(EnumSet.of(Flag.MOVE, Flag.LOOK)); } @Override public boolean canUse() { System.out.println("[DEBUG] LoopingGoal canUse => cooldownTimer=" + cooldownTimer); if (cooldownTimer > 0) { cooldownTimer--; return false; } BlockPos entityPos = entity.blockPosition(); boolean canUse = entity.isWaterAbove(entityPos, 4); System.out.println("[DEBUG] LoopingGoal canUse => WATER " + (canUse ? "DETECTED" : "NOT DETECTED") + " at " + entityPos + ", returning " + canUse); return canUse; } @Override public void start() { entity.setAnimation("looping"); animationTimer = 63; isLoopingActive = true; System.out.println("[DEBUG] Looping animation STARTED. Timer=" + animationTimer + ", gameTime=" + entity.level().getGameTime()); } @Override public boolean canContinueToUse() { System.out.println("[DEBUG] LoopingGoal canContinueToUse => animationTimer=" + animationTimer); return animationTimer > 0; } @Override public void tick() { animationTimer--; System.out.println("[DEBUG] LoopingGoal TICK => animationTimer=" + animationTimer); // We stop ONLY if we are still looping if (animationTimer <= 0 && isLoopingActive) { System.out.println("[DEBUG] condition => animationTimer <= 0 && isLoopingActive"); stop(); } } @Override public void stop() { // Check if already stopped if (!isLoopingActive) { System.out.println("[DEBUG] stop() called again, but isLoopingActive = false. Doing nothing."); return; } System.out.println("[DEBUG] Looping STOP at tick=" + entity.level().getGameTime() + ", last known rotation=" + entity.getXRot() + "/" + entity.getYRot() + ", animationTimer=" + animationTimer); // Immediately switch to "swim" entity.setAnimation("swim"); // Reset cooldown cooldownTimer = cooldownTime; // Disable looping to prevent a second stop isLoopingActive = false; System.out.println("[DEBUG] Looping STOP => setAnimation('swim'), cooldownTimer=" + cooldownTimer); } }  
    • So is the intention of the crusher for ores meant to be used with a silk touch pickaxe or something? Cause that seems like too much effort just to profit off of the machine, when everything drops as raw materials now. Am I just missing something? 
  • Topics

×
×
  • Create New...

Important Information

By using this site, you agree to our Terms of Use.