i have a gui system where if the player dosent touch a gui element it make them fall like physics . but the problem is everything is fine until you play the game for like 2 minutes and then it starts lagging insanly i figured the problem is that theres a for i,v loop inside of a while true loop . i tried to change it to a runservice. heartbeat but the problem is the same, i dont know how to fix it because it is neccecary for the for i,v loop to be there and the while true loop also. heres the script anyways this was possible using guicollisionservice made by the community
while true do
task.wait()
if Jumping == false then
for i,v in group:getColliders() do
if not GuiCollisionService.isColliding(player, v) then
falling = true
player.Position -= UDim2.new(0,0,0.0025,0)
else
falling = false
break
end
end
end
end
This is very inconsistent.
What your script is doing, is every frame your client surpasses, that while loop will activate, and not only it runs code inside, but that code has a for loop, which takes more power, resulting in lag.
The actual reason why it lags is because the game has to lower your framerate, so the while loop is more slower, considering it runs every frame.
If it checks if your player is colliding with something, then I’d use an event, probably using :GetPropertyChangedSignal()
whenever the player moves.
it says maximum re-entry for instance.changed exceeded
because you are checking if the player moved and if so then move it so its stuck in a loop
The solution I was about to type was similar to yours. Probably should scrap that.
My best option is to make the delay much longer, but short enough. It can be task.wait(0.2)
or shorter if you don’t experience lag.
but as i said i need to check every frame because i dont want the player floating for a couple of milllisecs
It doesn’t have to be every millisecond, you don’t have to worry about that. Some games make it quite literally 2-3 seconds.
Firstly this doesn’t run every frame. task.wait() waits for every CPU frame, which usually is below 3ms on modern CPUs.
But anyways, OP, you are doing this wrong. You dont usually use while loops in programming, expecially in games. You need to use a event for this.
To detect if a player moves, there’s no accurate way to do this.
humanoid.MoveDirection
Works but a player might be able to walk without changing direction, to not fire the changed event. And someone can be teleported without the client moving by keyboard or touchpad.
:GetPropertyChangedSignal("Position")
It would work but Roblox decides to not allow this for physics-related properties, for some reason.
You will need loops, but they must not be recurring so quickly, probably every split second.
this is what @DiamondDrencher was referring to
I’m assuming you’d want to check if the player’s position changes, based off of your code.
If not, change it
local co
player:GetPropertyChangedSignal("Position"):Connect(function()
if co then coroutine.yield(co) coroutine.close(co) co = nil end
co = coroutine.create(function()
if not Jumping then
for _, v in group:getColliders() do
if not GuiCollisionService.isColliding(player, v) then
falling = true
player.Position -= UDim2.new(0,0,0.0025,0)
else
falling = false
break
end
end
end
end)
coroutine.resume(co)
end)
based off of the code, it’s probably gui physics and not vector3s
UIs don’t have physics, the position is really just defined as a position on your screen.
like a 2d game, with gravity. I’m not necessarily referring to literal physics
but the problem is your stuck in a loop where the script checks if the player moves and if so then move down it so it gets stuck in a moving loop
I’ve edited it with coroutines.
sadly its still not working the player is still floating in the air it dosent go down