Jump to content

[1.8.9] [SOLVED] PlayerTickEvent not applying damage when it should


Daeruin

Recommended Posts

I'm working on a little feature that is supposed to apply damage to the player when they are walking or sprinting across a specific kind of block on a semi-random basis. As a test, I have created a PlayerTickEvent:

 

 

@SubscribeEvent
public void onPlayertick(PlayerTickEvent event) {

	ticksCounted += 1;

	Entity entity = event.player;
	BlockPos posBelowPlayer = new BlockPos((int) Math.floor(entity.posX), (int) Math.floor(entity.posY) - 0.2D, (int) Math.floor(entity.posZ));
	Block blockBelow = entity.worldObj.getBlockState(posBelowPlayer).getBlock();

	if (ticksCounted % 100 == 0
			&& entity instanceof EntityPlayer
			&& blockBelow == Blocks.dirt
			&& entity.onGround
			&& !entity.isSneaking()
			&& entity.worldObj.getDifficulty() != EnumDifficulty.PEACEFUL) {
		System.out.println("ALL CHECKS PASSED");
		event.player.attackEntityFrom(DamageSource.generic, 1.0F);
	}

 

 

As you can see, I'm not actually checking if the player is moving yet. I'm using dirt for now (because it's plentiful). I haven't applied any kind of randomization. I'm just trying to get it to happen every 100 ticks as a test. But it isn't happening. I'm keeping my player just standing there on dirt blocks, and can see my debug statement "ALL CHECKS PASSED" occurring every 100 ticks like clockwork, but the damage is only being applied every 20th time or so. It seems random. Any idea what is preventing the damage from being applied?

 

As a secondary question, how would you recommend checking if the player is walking or sprinting over the block? I was thinking that as long as the player is on the ground (event.entity.onGround == true) and not sneaking (event.entity.isSneaking() == true) and their X movement isn't 0 (event.entity.motionX != 0), then that's enough to apply the damage. I've implemented the first two, but not the last. Does that sound like it would work?

 

Oh, and a third, very minor, question. This is how I'm registering this event:

 

        FMLCommonHandler.instance().bus().register(new PlayerTickEventHandler());

 

But Eclipse is telling me that "bus()" is deprecated. It seems to be working as is, but I'm curious what is the correct way to do this for 1.8.9. I did some searching and couldn't find a quick answer, so I thought I'd tack it onto this post.

 

Thanks in advance!

 

Edit: Added clarification to subject title

Link to comment
Share on other sites

Further testing shows the event is only happening on the client side (event.player.worldObj.isRemote is always TRUE, and the event doesn't fire at all if I register it only in the server proxy). Does that mean I need to set up a packet handler to make sure the damage gets communicated to the server? I haven't delved into packet handling yet.

Link to comment
Share on other sites

OK, I kind of figured out what's causing this weirdness. To answer your question, I am registering the event handler in my Common Proxy during post initialization. I tried registering it in Server Proxy and Client Proxy, in pre, init, and post, just to see what would happen. It's interesting that the event doesn't fire at all when registered in Server Proxy, but I think that's a red herring.

 

I created a new event without all the extra stuff. All it does is increment the tick value and print to the console the tick value, phase, and whether the world is remote. I found that I get ticks from client and server in roughly equal numbers, but in irregular patterns—sometimes it's 2 ticks in a row from the server, then 2 from client, for 20 or 30 ticks; then I'll get 10 ticks in a row from the server and 10 from the client for a little while. So it's a toss of the dice whether any given tick is coming from the client or the server. Either way, the phase alternates every other tick regularly.

 

So next I set it up to check if the world is remote every X number of ticks. For example:

 

public class TickTest {

int tick;

@SubscribeEvent
public void onTick(PlayerTickEvent event) {
	tick += 1;
	if (tick % 50 == 0) {
		System.out.println("Tick: " + tick + " - Phase: " + (event.phase) + " - World remote? " + event.player.worldObj.isRemote);
		event.player.attackEntityFrom(DamageSource.generic, 1.0F);
	}
}
}

 

The code above has a tick firing from the server about 5 out of 15 times. I guess that's not too weird, given the uneven patterns I was seeing when checking every tick. I'm guessing it would have evened out to 50% of the time if I waited long enough. I also checked the tick phase, and it was always END, no matter what. Sample from log output:

 

 

[22:47:54] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 50 - Phase: END - World remote? true
[22:47:55] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 100 - Phase: END - World remote? true
[22:47:56] [server thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 150 - Phase: END - World remote? false
[22:47:56] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 200 - Phase: END - World remote? true
[22:47:57] [server thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 250 - Phase: END - World remote? false
[22:47:58] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 300 - Phase: END - World remote? true
[22:47:58] [server thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 350 - Phase: END - World remote? false
[22:47:59] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 400 - Phase: END - World remote? true
[22:48:00] [server thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 450 - Phase: END - World remote? false
[22:48:00] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 500 - Phase: END - World remote? true
[22:48:01] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 550 - Phase: END - World remote? true
[22:48:02] [Client thread/INFO] [sTDOUT]: [com.daeruin.primalcraft.events.TickTest:onTick:17]: Tick: 600 - Phase: END - World remote? true

 

 

Every time the tick comes from the server, the code to damage the player works (well, it takes a couple hundred ticks upon starting up the world before the damage starts registering). When the tick comes from the client, no damage is applied.

 

If I change the above code to check every 100 ticks, I get almost every single tick occurring on the client (in one test I got a single tick from the server after waiting 4000 ticks, in another test it took 2000 ticks). The reason my previous tests seemed to be occurring only on the client is because I was only checking every 100 ticks. If I change it to every 101 ticks, I get roughly every other one from the server.

 

So it appears if I check every X ticks, I just need to randomly pick the right value for X to get the code to execute regularly. Unless someone else can see a pattern that I can't.

Link to comment
Share on other sites

The player doesn't take any damage on the client because it's not supposed to - changes to data should only ever occur on the server, which then, if the data is important for the client to know about (e.g. for rendering GUIs), notifies the client of the current values.

 

Since you are testing on single player, the client and server are not really separated - the integrated server runs in the same program instance as the client - so your event handler only has one instance shared between the two. When you increment the tick counter on either side, the increased value is also seen on the other side, so both sides are incrementing the same counter. You would not notice this behavior when running on a dedicated server.

 

Anyway, to fix your issue, nest all of your event code in an 'if (!event.player.worldObj.isRemote)' statement - that will ensure your code only runs on the logical server side, which is what you want.

Link to comment
Share on other sites

Yes, I tried that after I posted as it seemed the next logical step. The behavior I'm seeing is that I can't predict whether any given tick is occurring on the server or the client. So if I check for whether the world is remote, THEN check whether a specific number of ticks have occurred (e.g., tickCount % 50), chances are good that specific tick was on client side (for example, see ticks 50 and 100 from the sample output in my last post), and nothing happens. In fact, when I'm checking every 100 ticks, the vast majority of those ticks are occurring on client side--embedding that kind of check inside a check for whether the world is remote will practically guarantee that the event code never occurs, especially after adding the other conditions I want to add (walking or sprinting on a specific kind of block).

 

Maybe instead of checking every 100 ticks, I need to just generate a random number that should let the code execute 1% of the time. Same net result, hopefully.

Link to comment
Share on other sites

If the tick is happening on the server side and the

Phase

is correct (it doesn't really matter whether it runs in

START

or

END

as long as you pick one), increment your tick counter and check if the appropriate amount of ticks have passed. If you don't check the side and

Phase

, the handler will be called twice per tick per side for every player.

 

You can't store the tick counter as a field of your event handler, since the same handler will be called for every player on the server. The more players there are, the more often one of them (there's no guarantee as to which one) will take damage. Event handlers are essentially singletons like

Block

s or

Item

s, so you can't store data for a specific object in their fields.

 

The

Entity#ticksExisted

field stores the number of ticks the entity has been alive for. Use this instead of maintaining your own tick counter.

Please don't PM me to ask for help. Asking your question in a public thread preserves it for people who are having the same problem in the future.

Link to comment
Share on other sites

Good point. I really haven't been thinking about server use at all, but if the event gets called every time ANY player ticks . . . yeah. Not quite what I was imagining.

 

So I check if the tick is happening on server side, in the correct phase, and whether Entity#ticksExisted % 100 is zero, then apply the damage. That sounds like it would be more reliable. I'll try it when I get home tonight.

 

I just realized that I probably don't need to check if the entity is a player, because I'm using PlayerTickEvent. In my earlier code I created an Entity variable named "entity" and assigned it the value of event.player, then later on I checked to see if "entity" is an instance of EntityPlayer. Seems a little redundant, no?

Link to comment
Share on other sites

So I check if the tick is happening on server side, in the correct phase, and whether Entity#ticksExisted % 100 is zero, then apply the damage. That sounds like it would be more reliable. I'll try it when I get home tonight.

That should work, yes.

 

I just realized that I probably don't need to check if the entity is a player, because I'm using PlayerTickEvent. In my earlier code I created an Entity variable named "entity" and assigned it the value of event.player, then later on I checked to see if "entity" is an instance of EntityPlayer. Seems a little redundant, no?

That is indeed redundant. As the name suggests,

PlayerTickEvent

is only fired for players.

Please don't PM me to ask for help. Asking your question in a public thread preserves it for people who are having the same problem in the future.

Link to comment
Share on other sites

Awesome, that worked perfectly!

 

However, now my check for movement is failing. It appears that whenever the world is not remote, event.player.motionX is always zero. Any other ideas how I can accomplish this? I need the event to trigger when the player is walking or sprinting across a certain type of block, and I need it to apply damage.

Link to comment
Share on other sites

Or you can compare the player's lastPos values with its pos values and see if they have changed.

Apparently I'm a complete and utter jerk and come to this forum just like to make fun of people, be confrontational, and make your personal life miserable.  If you think this is the case, JUST REPORT ME.  Otherwise you're just going to get reported when you reply to my posts and point it out, because odds are, I was trying to be nice.

 

Exception: If you do not understand Java, I WILL NOT HELP YOU and your thread will get locked.

 

DO NOT PM ME WITH PROBLEMS. No help will be given.

Link to comment
Share on other sites

Draco, thanks for the suggestion. I tried it out. Unfortunately, after making sure that the world is not remote and the phase is START (or END, doesn't make a difference), (event.player.posX == event.player.lastTickPosX) only seems to be false about one tick out of twenty, even when I'm moving constantly.

 

I'm trying to learn about packets so I can try coolAlias's suggestion. It's taking a while, but I'll try to report on that. In the meantime, any other suggestions are welcome.

Link to comment
Share on other sites

  • 2 weeks later...

So I spent a while learning about packets and set up a very basic thirst system while following some tutorials by coolAlias, diesieben, and jabelar. Cool stuff.

 

After doing all that, I changed my PlayerTickEvent to listen on the client side and created a packet (called PlayerToeStub, haha) to perform the damage on the server side. Unfortunately, it doesn't seem to work. I registered the packet as a server message and set up the handler to run on the correct thread (I think) but a check for world.isRemote inside the runnable shows that it still seems to be running on the client, and the damage doesn't get applied. I must be doing something wrong, but I don't know what it is. Any ideas or advice?

 

PlayerTickEvent (listens on client for the right conditions, then sends the packet)

 

public class PrimalPlayerTickEvent {

@SubscribeEvent
public void onTick(PlayerTickEvent event) {

	Entity entity = event.player;
	BlockPos posBelowPlayer = new BlockPos((int) Math.floor(entity.posX), (int) Math.floor(entity.posY) - 0.2D, (int) Math.floor(entity.posZ));
	Block blockBelow = entity.worldObj.getBlockState(posBelowPlayer).getBlock();

	if (entity.worldObj.isRemote) {
		if (event.phase == Phase.END) {
			if (entity.ticksExisted % 100 == 0) {
				System.out.println("Block below: " + blockBelow);
				if (blockBelow == Blocks.dirt || blockBelow == Blocks.grass || blockBelow == Blocks.stone || blockBelow == Blocks.gravel) {
					System.out.println("On ground: " + entity.onGround);
					if (entity.onGround) {
						System.out.println("Sneaking: " + entity.isSneaking());
						if (!entity.isSneaking()) {
							System.out.println("Moving: " + entity.motionX);
							if (entity.motionX != 0) {
								if (entity.worldObj.getDifficulty() != EnumDifficulty.PEACEFUL) {
									System.out.println("ALL CHECKS PASSED");
									PrimalPacketHandler.INSTANCE.sendToServer(new PrimalToeStubPacket(event.player));
	}}}}}}}}

}
}

 

 

Packet (sends damage to be applied to player on server)

 

public class PrimalToeStubPacket implements IMessage {

private int playerId;

public PrimalToeStubPacket() {}

public PrimalToeStubPacket(EntityPlayer player) {
	this.playerId = player.getEntityId();
}

@Override
public void fromBytes(ByteBuf buffer) {
	this.playerId = buffer.readInt();
}

@Override
public void toBytes(ByteBuf buffer) {
	buffer.writeInt(playerId);
}

    public static class PrimalToeStubHandler implements IMessageHandler<PrimalToeStubPacket, IMessage> {

	@Override
    public IMessage onMessage(final PrimalToeStubPacket message, final MessageContext ctx) {
        IThreadListener mainThread = (WorldServer) ctx.getServerHandler().playerEntity.worldObj;
        mainThread.addScheduledTask(new Runnable() {
            @Override
            public void run() {
                Entity entity = ctx.getServerHandler().playerEntity;
    			if (entity instanceof EntityPlayer)
    			{
    				entity.attackEntityFrom(DamageSource.generic, 1.0F);
    			}
            }
        });
        return null;
    }
    
    }

}

 

 

(Side note: I just realized that I'm creating a playerId variable but not using it anywhere—I think that's a holdover from the thirst system that I started to implement based on something I read. But I'm not using the playerId in my thirst packet, either, and everything seems to be working fine in single player mode. Maybe it will become problematic with multiple players?)

 

Packet registration in CommonProxy:

 

public class CommonProxy {

public void preInit(FMLPreInitializationEvent e) {

	PrimalPacketHandler.INSTANCE.registerMessage(PrimalThirstHandler.class, PrimalThirstPacket.class, 0, Side.CLIENT);
	PrimalPacketHandler.INSTANCE.registerMessage(PrimalToeStubHandler.class, PrimalToeStubPacket.class, 1, Side.SERVER);

}

 

Link to comment
Share on other sites

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

    • Dalam dunia perjudian online yang berkembang pesat, mencari platform yang dapat memberikan kemenangan maksimal dan hasil terbaik adalah impian setiap penjudi. OLXTOTO, dengan bangga, mempersembahkan dirinya sebagai jawaban atas pencarian itu. Sebagai platform terbesar untuk kemenangan maksimal dan hasil optimal, OLXTOTO telah menciptakan gelombang besar di komunitas perjudian online. Satu dari banyak keunggulan yang dimiliki OLXTOTO adalah koleksi permainan yang luas dan beragam. Dari togel hingga slot online, dari live casino hingga permainan kartu klasik, OLXTOTO memiliki sesuatu untuk setiap pemain. Dibangun dengan teknologi terkini dan dikembangkan oleh para ahli industri, setiap permainan di platform ini dirancang untuk memberikan pengalaman yang tak tertandingi bagi para penjudi. Namun, keunggulan OLXTOTO tidak hanya terletak pada variasi permainan yang mereka tawarkan. Mereka juga menonjol karena komitmen mereka terhadap keamanan dan keadilan. Dengan sistem keamanan tingkat tinggi dan proses audit yang ketat, OLXTOTO memastikan bahwa setiap putaran permainan berjalan dengan adil dan transparan. Para pemain dapat merasa aman dan yakin bahwa pengalaman berjudi mereka di OLXTOTO tidak akan terganggu oleh masalah keamanan atau keadilan. Tak hanya itu, OLXTOTO juga terkenal karena layanan pelanggan yang luar biasa. Tim dukungan mereka selalu siap sedia untuk membantu para pemain dengan segala pertanyaan atau masalah yang mereka hadapi. Dengan respon cepat dan solusi yang efisien, OLXTOTO memastikan bahwa pengalaman berjudi para pemain tetap mulus dan menyenangkan. Dengan semua fitur dan keunggulan yang ditawarkannya, tidak mengherankan bahwa OLXTOTO telah menjadi pilihan utama bagi jutaan penjudi online di seluruh dunia. Jika Anda mencari platform yang dapat memberikan kemenangan maksimal dan hasil optimal, tidak perlu mencari lebih jauh dari OLXTOTO. Bergabunglah dengan OLXTOTO hari ini dan mulailah petualangan Anda menuju kemenangan besar dan hasil terbaik!
    • Selamat datang di OLXTOTO, situs slot gacor terpanas yang sedang booming di industri perjudian online. Jika Anda mencari pengalaman bermain yang luar biasa, maka OLXTOTO adalah tempat yang tepat untuk Anda. Dapatkan sensasi tidak biasa dengan variasi slot online terlengkap dan peluang memenangkan jackpot slot maxwin yang sering. Di sini, Anda akan merasakan keseruan yang luar biasa dalam bermain judi slot. DAFTAR OLXTOTO DISINI LOGIN OLXTOTO DISINI AKUN PRO OLXTOTO DISINI   Jackpot Slot Maxwin Sering Untuk Peluang Besar Di OLXTOTO, kami tidak hanya memberikan hadiah slot biasa, tapi juga memberikan kesempatan kepada pemain untuk memenangkan jackpot slot maxwin yang sering. Dengan demikian, Anda dapat meraih keberuntungan besar dan memenangkan ribuan rupiah sebagai hadiah jackpot slot maxwin kami. Jackpot slot maxwin merupakan peluang besar bagi para pemain judi slot untuk meraih keuntungan yang lebih besar. Dalam permainan kami, Anda tidak harus terpaku pada kemenangan biasa saja. Kami hadir dengan jackpot slot maxwin yang sering, sehingga Anda memiliki peluang yang lebih besar untuk meraih kemenangan besar dengan hadiah yang menggiurkan. Dalam permainan judi slot, pengalaman bermain bukan hanya tentang keseruan dan hiburan semata. Kami memahami bahwa para pemain juga menginginkan kesempatan untuk meraih keberuntungan besar. Oleh karena itu, OLXTOTO hadir dengan jackpot slot maxwin yang sering untuk memberikan peluang besar kepada para pemain kami. Peluang Besar Menang Jackpot Slot Maxwin Peluang menang jackpot slot maxwin di OLXTOTO sangatlah besar. Anda tidak perlu khawatir tentang batasan atau pembatasan dalam meraih jackpot tersebut. Kami ingin memberikan kesempatan kepada semua pemain kami untuk merasakan sensasi menang dalam jumlah yang luar biasa. Jackpot slot maxwin kami dibuka untuk semua pemain judi slot di OLXTOTO. Anda memiliki peluang yang sama dengan pemain lainnya untuk memenangkan hadiah jackpot yang besar. Kami percaya bahwa semua orang memiliki kesempatan untuk meraih keberuntungan besar, dan itulah mengapa kami menyediakan jackpot slot maxwin yang sering untuk memenuhi harapan dan keinginan Anda.   Kesimpulan OLXTOTO adalah situs slot gacor terbaik yang memberikan pengalaman bermain judi slot online yang tak terlupakan. Dengan variasi slot online terlengkap dan peluang memenangkan jackpot slot maxwin yang sering, OLXTOTO menjadi pilihan terbaik bagi para pemain yang mencari kesenangan dan kemenangan besar dalam perjudian online. Di samping itu, OLXTOTO juga menawarkan layanan pelanggan yang ramah dan responsif, siap membantu setiap pemain dalam mengatasi masalah teknis atau pertanyaan seputar perjudian online. Kami menjaga integritas game dan memberikan lingkungan bermain yang adil serta menjalankan kebijakan perlindungan pelanggan yang cermat. Bergabunglah dengan OLXTOTO sekarang dan nikmati pengalaman bermain slot online yang luar biasa. Jadilah bagian dari komunitas perjudian yang mengagumkan ini dan raih kesempatan untuk meraih kemenangan besar. Dapatkan akses mudah dan praktis ke situs OLXTOTO dan rasakan sensasi bermain judi slot yang tak terlupakan.  
    • OLXTOTO: Platform Maxwin dan Gacor Terbesar Sepanjang Masa Di dunia perjudian online yang begitu kompetitif, mencari platform yang dapat memberikan kemenangan maksimal (Maxwin) dan hasil terbaik (Gacor) adalah prioritas bagi para penjudi yang cerdas. Dalam upaya ini, OLXTOTO telah muncul sebagai pemain kunci yang mengubah lanskap perjudian online dengan menawarkan pengalaman tanpa tandingan.     Sejak diluncurkan, OLXTOTO telah menjadi sorotan industri perjudian online. Dikenal sebagai "Platform Maxwin dan Gacor Terbesar Sepanjang Masa", OLXTOTO telah menarik perhatian pemain dari seluruh dunia dengan reputasinya yang solid dan kinerja yang luar biasa. Salah satu fitur utama yang membedakan OLXTOTO dari pesaingnya adalah komitmen mereka untuk memberikan pengalaman berjudi yang unik dan memuaskan. Dengan koleksi game yang luas dan beragam, termasuk togel, slot online, live casino, dan banyak lagi, OLXTOTO menawarkan sesuatu untuk semua orang. Dibangun dengan teknologi terkini dan didukung oleh tim ahli yang berdedikasi, platform ini memastikan bahwa setiap pengalaman berjudi di OLXTOTO tidak hanya menghibur, tetapi juga menguntungkan. Namun, keunggulan OLXTOTO tidak hanya terletak pada permainan yang mereka tawarkan. Mereka juga terkenal karena keamanan dan keadilan yang mereka berikan kepada para pemain mereka. Dengan sistem keamanan tingkat tinggi dan audit rutin yang dilakukan oleh otoritas regulasi independen, para pemain dapat yakin bahwa setiap putaran permainan di OLXTOTO adalah adil dan transparan. Tidak hanya itu, OLXTOTO juga dikenal karena layanan pelanggan yang luar biasa. Dengan tim dukungan yang ramah dan responsif, para pemain dapat yakin bahwa setiap pertanyaan atau masalah mereka akan ditangani dengan cepat dan efisien. Dengan semua fitur dan keunggulan yang ditawarkannya, tidak mengherankan bahwa OLXTOTO telah menjadi platform pilihan bagi para penjudi online yang mencari kemenangan maksimal dan hasil terbaik. Jadi, jika Anda ingin bergabung dengan jutaan pemain yang telah merasakan keajaiban OLXTOTO, jangan ragu untuk mendaftar dan mulai bermain hari ini!  
    • OLXTOTO adalah bandar slot yang terkenal dan terpercaya di Indonesia. Mereka menawarkan berbagai jenis permainan slot yang menarik dan menghibur. Dengan tampilan yang menarik dan grafis yang berkualitas tinggi, pemain akan merasa seperti berada di kasino sungguhan. OLXTOTO juga menyediakan layanan pelanggan yang ramah dan responsif, siap membantu pemain dengan segala pertanyaan atau masalah yang mereka hadapi. Daftar =  https://surkale.me/Olxtotodotcom1
    • DAFTAR & LOGIN BIGO4D   Bigo4D adalah situs slot online yang populer dan menarik perhatian banyak pemain slot di Indonesia. Dengan berbagai game slot yang unik dan menarik, Bigo4D menjadi tempat yang ideal untuk pemula dan pahlawan slot yang berpengalaman. Dalam artikel ini, kami akan membahas tentang Bigo4D sebagai situs slot terbesar dan menarik yang saat ini banyak dijajaki oleh pemain slot online.
  • Topics

×
×
  • Create New...

Important Information

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