Jump to content
  • 0

Magic Gem Interlude Acis 401 Help


Question

Posted

hello i found the magic gem code from the source l2jtenkai and i wanted to try it in interlude but i have some problems I don't know how to make it only be used once to avoid spam and if the player moves the npc disappear I leave a video for you to see

 

https://www.youtube.com/watch?v=BRpk92UzuLc&feature=youtu.be

 

Item 

MagicGem.java

package net.sf.l2j.gameserver.handler.itemhandlers;

import net.sf.l2j.A.Instance.InstanceManager;
import net.sf.l2j.gameserver.data.xml.SummonItemData;
import net.sf.l2j.gameserver.enums.ZoneId;
import net.sf.l2j.gameserver.handler.IItemHandler;
import net.sf.l2j.gameserver.model.actor.Playable;
import net.sf.l2j.gameserver.model.actor.Player;
import net.sf.l2j.gameserver.model.holder.IntIntHolder;
import net.sf.l2j.gameserver.model.item.instance.ItemInstance;
import net.sf.l2j.gameserver.network.SystemMessageId;

public class MagicGem implements IItemHandler {


    @Override
    public void useItem(Playable playable, ItemInstance item, boolean forceUse) {
        if (!(playable instanceof Player)) {
            return;
        }

        Player player = (Player) playable;
        
     
		if (player.isSitting())
		{
			player.sendPacket(SystemMessageId.CANT_MOVE_SITTING);
			return;
		}
		if (player.isMoving()) 
		{
			player.sendPacket(SystemMessageId.CANNOT_EXCHANCE_USED_ITEM);
			return;
		}
		
		if (player.isInObserverMode())
			return;
		
		if (player.isAllSkillsDisabled() || player.getCast().isCastingNow())
			return;
		
		final IntIntHolder sitem = SummonItemData.getInstance().getSummonItem(item.getItemId());
		
		if ((player.getSummon() != null || player.isMounted()) && sitem.getValue() > 0)
		{
			player.sendPacket(SystemMessageId.SUMMON_ONLY_ONE);
			return;
		}
		
		if (player.getAttack().isAttackingNow())
		{
			player.sendPacket(SystemMessageId.YOU_CANNOT_SUMMON_IN_COMBAT);
			return;
		}

        if (player.getInstanceId() == 0 &&
                !player.isInsideZone(ZoneId.PVP) &&
                (!player.isInsideZone(ZoneId.NO_SUMMON_FRIEND) ||
             
                !player.isInOlympiadMode() &&
                InstanceManager.getInstance().getInstance(player.getObjectId()) == null && player.getPvpFlag() == 0)){
  
        	player.spawnServitors();          
            player.sendMessage("You use a Magic Gem.");

        } else {
        	
            player.sendMessage("You cannot use a Magic Gem right now.");
        }
        
        
        return;
    }
    

    
}

 

 

Player.java

	public void spawnServitors() {
		_instance = InstanceManager.getInstance().createInstance();
		Spawn servitor;

		float angle = Rnd.get(1000);
		int sCount = 4;
		int lifeTime = 60;
		for (int i = 0; i < sCount; i++) {
			servitor = getNpcServitor(i);
			if (servitor != null) {
				servitor.setInstanceId(_instance.getId());
				servitor.setLoc(Math.round(getX() + (float) Math.cos(angle / 1000 * 2 * Math.PI) * 30),
						Math.round(getY() + (float) Math.sin(angle / 1000 * 2 * Math.PI) * 30),
								getZ() + 75, i);
				int heading = (int) Math
						.round(Math.atan2(getY() - servitor.getLocY(), getX() - servitor.getLocX()) / Math.PI * 32768);
				if (heading < 0) {
					heading = 65535 + heading;
				}
				servitor.setHeading(heading);

					if (_instance != null) {
						final Npc npc = servitor.doSpawn();
						npc.scheduleDespawn(lifeTime * 1000L);
				}

			}
			angle += 1000 / sCount;
		}
	}
	
	// Magic Gem
	private Spawn[] _npcServitors = new Spawn[4];

	public Spawn getNpcServitor(int id) {
		if (_npcServitors[id] != null) {
			return _npcServitors[id];
		}
		Spawn spawn = null;

		try {
			NpcTemplate tmpl;
			switch (id) {
				case 0:
					tmpl = NpcData.getInstance().getTemplate(7);
					break;
				case 1:
					tmpl = NpcData.getInstance().getTemplate(50006);
					break;
				case 2:
					tmpl = NpcData.getInstance().getTemplate(30080);
					break;
				default:
					tmpl = NpcData.getInstance().getTemplate(30103);
			}
			spawn = new Spawn(tmpl);
		} catch (Exception e) {
			e.printStackTrace();
		}
		_npcServitors[id] = spawn;
		return _npcServitors[id];
	}	

 

 

I'd appreciate your help

9 answers to this question

Recommended Posts

  • 0
Posted
27 minutes ago, An4rchy said:

You want the npcs to disappear after the player moves?

 

Or make the old npcs disappear when the player uses it again?

the first thing to disappear when the player moves

  • 0
Posted

You need to attach the spawned NPCs to the Player first.

Then, you need to modify the packets MoveToLocation and MoveToPawn. When the packet is called (which means the player requested to move), you need to find those attached NPCs and delete them.

 

If you need further help let me know.

  • 0
Posted
1 hour ago, An4rchy said:

You need to attach the spawned NPCs to the Player first.

Then, you need to modify the packets MoveToLocation and MoveToPawn. When the packet is called (which means the player requested to move), you need to find those attached NPCs and delete them.

 

If you need further help let me know.

Is there something in acis to guide me in attaching the npc to the player?

  • 0
Posted (edited)

You need to store the Npc that is being created by the Spawn here (or the Spawn itself, same thing):

 

final Npc npc = servitor.doSpawn();

 

There's multiple ways to store them, a generic way could be to make a List<Npc> in Player and add the npc that is spawned to the list:

 

ArrayList<Npc> _gemSpawns = new ArrayList<>();

 

To add the npc to the list:

 

_gemSpawns.add(npc);

 

Then to delete the npcs:

 

for (Npc npc : _gemSpawns)
{
	Spawn spawn = npc.getSpawn();
	if (spawn != null)
	{
		spawn.setRespawnState(false);
		SpawnTable.getInstance().deleteSpawn(spawn, true);
	}
	npc.deleteMe();
}
_gemSpawns.clear();

 

 

HOWEVER, as I see in your code, there is a delete being scheduled for these npcs already. So you might run into NPE if you don't cancel the task to delete the npcs when the player moves. I don't know what the context of scheduleDespawn() is to help you out with that.

 

GL.

Edited by An4rchy
  • Like 1
  • 0
Posted
39 minutes ago, An4rchy said:

You need to store the Npc that is being created by the Spawn here (or the Spawn itself, same thing):

 

final Npc npc = servitor.doSpawn();

 

There's multiple ways to store them, a generic way could be to make a List<Npc> in Player and add the npc that is spawned to the list:

 

ArrayList<Npc> _gemSpawns = new ArrayList<>();

 

To add the npc to the list:

 

_gemSpawns.add(npc);

 

Then to delete the npcs:

 

for (Npc npc : _gemSpawns)
{
	Spawn spawn = npc.getSpawn();
	if (spawn != null)
	{
		spawn.setRespawnState(false);
		SpawnTable.getInstance().deleteSpawn(spawn, true);
	}
	npc.deleteMe();
}
_gemSpawns.clear();

 

 

HOWEVER, as I see in your code, there is a delete being scheduled for these npcs already. So you might run into NPE if you don't cancel the task to delete the npcs when the player moves. I don't know what the context of scheduleDespawn() is to help you out with that.

 

GL.

Thanks for the help I will try to do it and now it is only that after 60 seconds the npc will be deleted by themselves

  • 0
Posted
3 minutes ago, tensador27 said:

Thanks for the help I will try to do it and now it is only that after 60 seconds the npc will be deleted by themselves

 

You need to cancel this task when the player moves as well in order go avoid getting NPE (error). Save the Future locally and use Future.cancel() when you despawn the npcs on player movement.

  • 0
Posted (edited)

For the automatic despawn :

 

When you create the Spawn and once you spawned the Npc, you can define the despawn delay with npc.scheduleDespawn(despawnDelay); (which will generate a despawn task, there is no way to cancel it since it's not tracked)

 

----

 

For the move part :

 

You have to keep either a Spawn or Npc List/Set in Player, as suggested Anarchy.

 

If you want a guaranteed disappear on any type of move (even during a move), you have to write your disappear content on PlayerMove#updatePosition. Simply write a !_actor.getWhateverSpawns().isEmpty() check, for loop your list and despawn NPCs.

 

Be sure to use a concurrent list/set (prefer ConcurrentHashMap.newKeySet over CopyOnWriteArrayList due to performance), and not an arraylist (to avoid CME, which probably will occur if you decide to use item while moving).

 

-----

 

You also probably want to delete "old" Npcs before summoning new ones. Use same content than your PlayerMove#updatePosition check in the item handler before #spawnServitors call, to avoid to get billions of random NPCs.

Edited by Tryskell
  • 0
Posted
On 6/24/2022 at 1:48 PM, Tryskell said:

For the automatic despawn :

 

When you create the Spawn and once you spawned the Npc, you can define the despawn delay with npc.scheduleDespawn(despawnDelay); (which will generate a despawn task, there is no way to cancel it since it's not tracked)

 

----

 

For the move part :

 

You have to keep either a Spawn or Npc List/Set in Player, as suggested Anarchy.

 

If you want a guaranteed disappear on any type of move (even during a move), you have to write your disappear content on PlayerMove#updatePosition. Simply write a !_actor.getWhateverSpawns().isEmpty() check, for loop your list and despawn NPCs.

 

Be sure to use a concurrent list/set (prefer ConcurrentHashMap.newKeySet over CopyOnWriteArrayList due to performance), and not an arraylist (to avoid CME, which probably will occur if you decide to use item while moving).

 

-----

 

You also probably want to delete "old" Npcs before summoning new ones. Use same content than your PlayerMove#updatePosition check in the item handler before #spawnServitors call, to avoid to get billions of random NPCs.

 

hello, sorry, the nuisance, I managed to make the npc only see the one who calls them and that they disappear 60 seconds after using the item, I also made it so that they cannot spam the item but I can't get the npc to be automatically deleted when moving the player because I don't know where the player's movement is

 

this is how spawnservitors is

 

	public void spawnServitors()
	{
		/** The _player. */
		final L2PcInstance _player;
		_player = L2PcInstance.this;
		
		InstanceManager.getInstance().createInstance(getObjectId());
		L2Spawn servitor;
		float angle = Rnd.get(1000);
		int sCount = 4;
		
		for (int i = 0; i < sCount; i++)
		{
			final L2Spawn npc = servitor = getNpcServitor(i);
			if (servitor != null)
			{
				servitor.setInstanceId(getObjectId());
				_player.setInstanceId(_player.getObjectId());
				_player.getAppearance().setInvisible();
				servitor.setLocx(Math.round(getX() + (float) Math.cos(angle / 1000 * 2 * Math.PI) * 30));
				servitor.setLocy(Math.round(getY() + (float) Math.sin(angle / 1000 * 2 * Math.PI) * 65));
				servitor.setLocz(getZ());
				int heading = (int) Math.round(Math.atan2(getY() - servitor.getLocy(), getX() - servitor.getLocx()) / Math.PI * 32768);
				if (heading < 0)
				{
					heading = 65535 + heading;
				}
				servitor.setHeading(heading);
				
				if (InstanceManager.getInstance().getInstance(getObjectId()) != null)
				{
					servitor.doSpawn();
					ThreadPoolManager.getInstance().scheduleGeneral(() -> npc.getLastSpawn().decayMe(), 60000);
				}
				
				if (_player.isMoving() || !isInsideRadius(_player, Config.NPC_MAGIC_GEM_RADIUS, false, false))
				{
					ThreadPoolManager.getInstance().scheduleGeneral(() -> npc.getLastSpawn().decayMe(), 1000);
					_player.setInstanceId(0);
				}
			}
			angle += 1000 / sCount;
		}
	}

 

and this is the part of the item

 

 

		if (!player.getFloodProtectors().getMagicGem().tryPerformAction("Magic Gem"))
		{
			return;
		}
		
		if (!GrandBossManager.getInstance().checkIfInZone(null, player) && player.getInstanceId() == 0 && !player.isInsideZone(ZoneId.ZONE_PVP) && (!player.isInsideZone(ZoneId.ZONE_NOSUMMONFRIEND) || !player.isInsideZone(ZoneId.ZONE_TOWN)) && !player.isInOlympiadMode()
			&& !AttackStanceTaskManager.getInstance().getAttackStanceTask(player) && InstanceManager.getInstance().getInstance(player.getObjectId()) == null && player.getPvpFlag() == 0)
		{
			
			player.spawnServitors();
			
			player.sendMessage("You use a Magic Gem.");
			
			ThreadPoolManager.getInstance().scheduleGeneral(new Runnable()
			{
				@Override
				public void run()
				{
					player.setInstanceId(0);
					player.getAppearance().setVisible();
				}
			}, 59000);
		}
		else
		{
			player.sendMessage("You cannot use a Magic Gem right now.");
		}

 

Create an account or sign in to comment

You need to be a member in order to leave a comment

Create an account

Sign up for a new account in our community. It's easy!

Register a new account

Sign in

Already have an account? Sign in here.

Sign In Now


  • Posts

    • First, you need to understand what you're doing and what you want to achieve. You have to choose a server core. After that, decide what you want your server to include, code it, modify the client to fit your server, go public, and drink champagne.   If you know how to code, creating a server is relatively easy — a few months of work and you can make it happen. Modifying the client is a completely different story. There’s a lack of tutorials, tools, and source materials. I’m currently working on the client myself, and I’ve already spent over three weeks just trying to get started due to the lack of information. If you don’t have the knowledge and experience, you’ll need a team and a bag of money — but realistically, it just won’t succeed.
    • The server has been online and stable for over 2 months now, and we’re still going strong! No wipes, no shortcuts ~ just continuous work, daily fixes, events, and improvements to ensure the best possible experience.   Great News! 🔥 CHAPTER II IS COMING — GRACIA FINAL 🔥 On February 16, L2Elixir enters a new era. The server will be officially updated to Gracia Final, opening Chapter II of our journey. Expect new content, improvements, and surprises that will refresh the gameplay while keeping the classic Gracia Final spirit alive.   More challenges, more competition, and more reasons to log in.   📅 Update Date: February 16 ⚔️ Chapter II: Gracia Final This is not a reset. This is evolution.   Prepare yourselves — Chapter II begins soon.   Website: https://l2elixir.org/ Discord: https://discord.gg/5ydPHvhbxs    
    • Server owners, Top.MaxCheaters.com is now live and accepting Lineage 2 server listings. There is no voting, no rankings manipulation, and no paid advantages. Visibility is clean and equal, and early listings naturally appear at the top while the platform grows. If your server is active, it should already be listed. Submit here 👉https://Top.MaxCheaters.com This platform is part of the MaxCheaters.com network and is being built as a long-term reference point for the Lineage 2 community. — MaxCheaters.com Team
    • Hello! We are Genesis, small team that works on new Lineage 2 project. Our goal with this project is to create a fresh new place to play — built around real community feedback, with no aggressive pay-to-win donations and with carefully thought-out quality-of-life improvements, balance changes etc. We believe that even tho we all love this game, everyone has at least one or two things they would like to change in the game to make it more enjoyable. Thats why we want the comunity feedback to shape our server. Main information about the server: • Interlude Classic version • Rates: EXP x4 SP x2 Loot x2, Spoil x2 (not set in stone, might be changed) • Local & Server-Side Dualbox Protection • Complete, Clear Website with Integrated Account Panel (Game account creation, direct communication with support, bug reporting, voting and reward system) • Launcher – External Game Login System: manage all your accounts inside the launcher, “Play” button logs you directly into the game server Here are list of few changes we already added/decided to add to the server: • Reworked Client to fit interlude Era with upgraded Classic Ui • Custom Antibot system • Custom AntiDualBox System • Offline shops • Offline shop with buffs (available only in towns) • Mass Sweeper added to the game • Newbie buffs available all the way to lvl 76 (nothing crazy, but its free) • Slight balance change to Destroyer damage with Polearm and Cancel spell from SPS • PvP zones on every Epic spawn spot • Overbuffing blocked • And more! Since we put big focus on community feedback and suggestions, we are looking for people for our internal tests, that will discuss whether current changes „fit” into the game and maybe suggest some changes themselves. If what you’ve just read sounds interesting to you, if you want to help creating server fitted for you, join our server Discord. Help us to understand what Lineage 2 players in 2026 actually expect and need — so we can meet those expectations and avoid becoming just another server that dies a natural death.     Even if you’re not interested in playing right now, but you are a long-time Lineage 2 player, feel free to join our community. We would greatly appreciate your experience and feedback to help us improve and develop our project. Join the growing L2Genesis community: https://discord.gg/mcuHsQzNCm Also check our website: https://l2genesis.com/
  • Topics

×
×
  • Create New...

Important Information

This community uses essential cookies to function properly. Non-essential cookies and third-party services are used only with your consent. Read our Privacy Policy and We have placed cookies on your device to help make this website better. You can adjust your cookie settings, otherwise we'll assume you're okay to continue..