Could I get some help making a 3D player movement system?
Frodev Hello there, recently I decided making 2D games was way to much work for me (especially with 2D animations) that I chose to swap to doing 3D because unlike 2D, I know what Im doing (art wise), I then realized I have never made a 3D game in godot before and for the life of me, cannot figure out how to make player movement, I've looked at over 12 tutorials now and I've gotten no where.
So here I am!
If I may approach this from a different angle...
If this is a question of comfort(as in confidence in your skillset) and you mostly have experience with 3D when it comes to art and 2D when it comes to logic systems design... have you considered giving a game with prerendered 3D graphics a try?
But I also think it's not a bad idea to expand your knowledge a bit with programming for 3D. As @xyz said, it's just one extra axis. Albeit there is a bit of a complexity increase with regards to things like gimbal lock. Though as an experienced 3D artist that shouldn't be an alien concept to you.
I recommend https://docs.godotengine.org/en/4.0/getting_started/first_3d_game/index.html
This covers pretty much the basics for 3d character movement and jumping.
The Godot documentation is really helpful most of the time.
- Edited
kuligs2 Are you serious? What do you mean not work? Im devving in godot in 3d, and how come i found tutorials that "WORKS"?
I think its cause I'm probably just bad and broke it looking back on it, but the tutorials didnt really explain it really well so I couldnt fix it.
Ps: I've tried this tutorial and it ended up with me only being able to move left and right, and that the gravity thing didnt work at all, but I'll try it again
Megalomaniak
I never thought of this before! That is actually a pretty cool idea, sometime I'll have to try it out! Thanks!
trizZzle
I didnt realize they made the documents with all of these wonderful tutorials! Ill probably end up trying this cause well its in the docs, as long as it isnt outdated it should work (and sense the code is right there I shouldnt manage to break it), not to mention I love reading the docs!
Thanks!
- Edited
trizZzle
Ok the tutorials really nice! I got the movement for the most part ready to go! One problem I'm facing is the camera, I want to make in first person so I've followed some tutorials and again I seem to be breaking it (or maybe they are out dated) right now I do have some code, it doesnt seem to work and for the life of me I dont know why (cause I dont know how it would work but here goes!)
func _unhandled_input(event):
if event is InputEventMouseButton : # lock camera and what not
Input.mouse_mode = Input.MOUSE_MODE_CAPTURED
elif event.is_action_pressed('ui_cancel') :
Input.mouse_mode = Input.MOUSE_MODE_VISIBLE
if Input.mouse_mode == Input.MOUSE_MODE_CAPTURED and not GameValues.inMenu : # make sure we are in first person and not in a menu
if event is InputEventMouseMotion : # if the event is mouse motion, rotate!
neck.rotate_y(-event.relative.x * 0.01)
camera.rotate_x(-event.relative.y * 0.01)
camera.rotation.x = clamp(camera.rotation.x,deg_to_rad(-30),deg_to_rad(60))
now from doing some debugging I can say that whatever InputEventMouseMotion is never triggered, ever.
Although any code before it is indeed triggering so something goes wrong there, as to how to fix it I'm early in learning this stuff and so far have not been able to figure it out. So your help would be nice, thanks!
Thanks!
- Edited
don't use an elif
for the ui_cancel just a regular if.
func _unhandled_input(event):
if event is InputEventMouseButton : # lock camera and what not
Input.mouse_mode = Input.MOUSE_MODE_CAPTURED
if event.is_action_pressed('ui_cancel') :
if Input.mouse_mode == Input.MOUSE_MODE_CAPTURED:
Input.mouse_mode = Input.MOUSE_MODE_VISIBLE
if Input.mouse_mode == Input.MOUSE_MODE_CAPTURED:
print("mouse is captured")
if not GameValues.inMenu : # make sure we are in first person and not in a menu
print("passed through inMenu related check")
if event is InputEventMouseMotion : # if the event is mouse motion, rotate!
neck.rotate_y(-event.relative.x * 0.01)
camera.rotate_x(-event.relative.y * 0.01)
camera.rotation.x = clamp(camera.rotation.x,deg_to_rad(-30),deg_to_rad(60))
I'd probably also not put the checks for MOUSE_MODE_CAPTURED and GameValues.inMenu both onto the same line. Right now you have a hard time telling which of those checks causes you problems.
edit: Actually thinking about it I can see why you wanted to use elif for the ui_cancel but in that case maybe its worth keeping track of menu state and use a match based on that, while also being able to use the same state variable check also for rest of the relevant code...
Megalomaniak
Ok so I tried the code you suggested (PS: thanks for the pointers) but I have some problems, I realized that this only updates anytime I press a button and not the mouse actually moving, probably because its locked in the center of the screen and doesn't move, therefore the event is never mouse motion, so I was going to plug it into a process function but I realized that the when turning the camera:
Megalomaniak neck.rotate_y(-event.relative.x * 0.01)
camera.rotate_x(-event.relative.y * 0.01)
I use the event variable that was defined inside of func _unhandled_input(event): of course in a process function I cant get this (well to my knowledge) so what can I do instead? Cause I have no idea what I can use instead, probably because I for the most part dont even know what it does!
So yup
If you know how to fix it and would lend me your knowledge that would be great! If you need anything else about my project like info or something, I'm more than happy to provide.
Thanks again!
- Edited
Frodev I realized that this only updates anytime I press a button and not the mouse actually moving
The _unhandled_input(event): is just one of the loops for doing input stuff. You should maybe try the normal _input(event): loop too. Some things will work better in one than the other.
Frodev I use the event variable that was defined inside of func _unhandled_input(event): of course in a process function I cant get this (well to my knowledge) so what can I do instead?
Pretty sure you can call on the Input and InputEvent classes from process loops, so you might be able to make that work for you. For an example, read the velocity property from the InputEventMouseMotion class.
Megalomaniak
Ok I got it working! Thanks so much!
Megalomaniak
Hi there quick question, I just realized that the direction you move in is not based on the camera direction.
I've tried a couple of things, I was gonna try vector3.Forward and just use the camera position until I realize that from my knowledge, it isnt possible (or well I cant figure it out it probably is possible but what would I know).
So here I am asking for help!
How I have it now the camera is rotated only on the X and Z access where as the Y is rotated in another node, this node being the parent of the camera but the child of the player:
Anyhow this is how I am currently moving the player:
func _physics_process(delta):
# movement -----------------------------------------------------------------
var direction = Vector3.ZERO
if Input.is_action_pressed('WalkRight') :
direction.x += 1
if Input.is_action_pressed('WalkLeft') :
direction.x -= 1
if Input.is_action_pressed('WalkBackwards') :
direction.z += 1
if Input.is_action_pressed('WalkForward') :
direction.z -= 1
if is_on_floor() and Input.is_action_just_pressed("Jump") :
targetVelocity.y = jumpImpulse
if direction != Vector3.ZERO :
direction = direction.normalized()
$Pivot.look_at(position + direction, Vector3.UP)
targetVelocity.x = (direction.x * speed) * GameValues.gameSpeed
targetVelocity.z = (direction.z * speed) * GameValues.gameSpeed
if not is_on_floor() :
targetVelocity.y = (targetVelocity.y - (fallAcceleration * delta)) * GameValues.gameSpeed
velocity = targetVelocity
move_and_slide()
As we can see I just add to either the X or the Z or whatever, but obviously that isnt going to work but I currently dont know a way to do otherwise, if you do know a way please tell me, I've already spent 30+ minutes researching this topic but nothing so far.
Thanks so much!
Do you understand the difference between local space and global space transforms?
Megalomaniak
A little bit yes, I believe I understand the basics of it.
Also, what node is the script attached to? You are likely mostly dealing with local space values of nodes here and should perhaps instead use global coordinate space values in certain cases instead. And be careful of mixing local and global space values unless you know you need to do so.
Megalomaniak
The script is attached to the node called player:
I see what you are saying about using global positions, I assumed that for some reason just use move_and_slide() would do that on the global side.
But out of curiosity why does moving it using global positions do anything really? Is this gonna make it go in the direction or the camera? Cause I feel like it wouldnt but I dont know if thats how that works so here I am!