MirceaKitsune Moving a character body on a sphere surface with proper planetary gravity is not a big deal. It's just some linear algebra and some high school physics calculations.

For rigid bodies, you'd need to apply a gravity impulse each frame, with a direction pointing towards the planet center and a magnitude proportional to the distance. Also no big deal.

Navmeshes don't care about orientation or slopes. They only care about mesh connectivity/topology. You can wrap a navmesh onto a sphere surface without problems.

    xyz That's very good to hear, indeed my shader based approach is highly overkill then. Still I remember trying this once and nothing I did got the character to work right: If you or anyone else knows, so I don't open another thread just for this, could you point me to a simple GDScript example please? I just need to see how you properly orient a basic capsule character so their feet always point to '0 0 0' origin, apply gravity relative to that, WASD movement and mouse look working relative to this rotation.

    • xyz replied to this.

      MirceaKitsune I don't know if there's an exact example of this. I generally don't follow any tutorial content though. There might be something out there, maybe someone else knows.

      But imho, if you can't implement this initial setup on your own, you should perhaps start with a less tricky project because things will get even harder as you develop it further.

      MirceaKitsune Ok, here's your example. You owe us at least a small jam game based on this now 😉


      extends CharacterBody3D
      
      const SPEED = 5.0
      const JUMP_VELOCITY = 4.5
      const  GRAVITY = 20.0
      
      var vel_vertical = 0.0
      
      
      func _input(e):
      	if e is InputEventMouseMotion:
      		rotate_object_local(Vector3.UP, -e.relative.x * .02) # left-right turn
      
      
      func _physics_process(delta):
      	
      	# handle vertical velocity
      	if is_on_floor():
      		vel_vertical = 0.0
      	else:
      		vel_vertical -= GRAVITY * delta
      	
      	# jump
      	if Input.is_action_just_pressed(&"ui_accept"):
      		vel_vertical = JUMP_VELOCITY
      	
      	# horizontal movement velocity
      	var input_dir = Input.get_vector(&"ui_left", &"ui_right", &"ui_up", &"ui_down")
      	velocity = (global_basis.x * input_dir.x + global_basis.z * input_dir.y) * SPEED
      		
      	# add vertical velocity to final velocity
      	velocity += global_basis.y * vel_vertical
      	
      	move_and_slide()
      	
      	# orient on sphere
      	global_basis.y = global_position.normalized()
      	global_basis.x = global_basis.y.cross(global_basis.z)
      	global_basis.z = global_basis.x.cross(global_basis.y)
      	global_basis = global_basis.orthonormalized()
      	# set the up direction so move_and_slide is not confused
      	up_direction = global_basis.y

        xyz Thank you, I appreciate it! Can't make promises since I tend to start so many projects and rarely finished any, but should at least give it a good try. At least I can hopefully create a dynamic planet with a lot of detail which was my original goal.

        • xyz replied to this.

          MirceaKitsune You should at least try setting this up now and report back if it works. Otherwise I'll feel I typed this script in vain. Better yet, try to understand how it works.

            xyz dont worry, some people will find this usefull!, Thanks

            xyz Otherwise I'll feel I typed this script in vain.

            What you do is never in vain.

            • xyz replied to this.

              Tomcat What you do is never in vain.

              Hopefully 😃

              xyz Will do, I should try at least a small project at some point. At very least it will help others looking for answers to this exact question. But obviously I asked because I plan on using it just need to find the right setup with everything else.

              • xyz replied to this.

                MirceaKitsune Setting this up would require less effort than typing your original post. Not telling you what to do but since you went to some length to solicit the example code, it'd seem reasonable to at least try to run it.

                  xyz I plan to. I'm actually thinking what the best way to attempt simulating either a heightmap or voxel planet would be, without having to use one of the terrain plugins which are designed to be flat and most require C# to run: Was thinking of (ab)using CSG spheres but what I'm thinking of would be so slow it will likely just freeze.

                  • xyz replied to this.

                    xyz The plan is to go for acceptable realism, not ridiculously small but obviously not life sized either. If the player is 2m tall, I'd need to have the sphere at very least 1.000 units large and see how that works, after that see if I can push it to 10.000 and beyond.

                    Obviously it will need some form of LOD, which is harder to do out of square patches for a bent sphere unlike a flat ground, another reason why I initially wanted a deform shader for roundness. Currently my idea is to use noise to add CSG spheres based on camera distance, larger units / spheres and thus lower LOD the further away something is; I still risk having to use thousands of them even so, and to my knowledge the CSG system doesn't like so many items which cause slowness and random tares in the surface.

                    It would be a lot easier if Godot had a tesselation shader, but as it stands you can't add or remove vertices through GLSL only displace them. There's also collisions: Those can only be calculated on the CPU to my knowledge, you can't have a collision mesh read the displacements from a shader which would have been really amazing.

                    • xyz replied to this.

                      MirceaKitsune If the planet is only seen from player's perspective, then there's no point in making it spherical, or at least the sphere can be approximated using planar patches.

                      4 days later
                      8 days later

                      Sorry for the delay. I integrated your code with my old player script and am happy to say it works wonderfully! I managed to simplify it a bit so this is the only transform required for the base math:

                      global_basis.y = global_position.normalized()
                      global_basis.x = global_basis.y.cross(global_basis.z)
                      global_basis = global_basis.orthonormalized()

                      Here's the full player script, it contains more features including distance scaling and inverted gravity for inside-out planets or rings:

                      extends CharacterBody3D
                      
                      @export var Sensitivity_X = 0.01
                      @export var Sensitivity_Y = 0.01
                      @export var Minimum_Y_Look = -90
                      @export var Maximum_Y_Look = 90
                      @export var Accelaration = 100
                      @export var Decelaration = 25
                      @export var Air_Accelaration = 50
                      @export var Air_Decelaration = 5
                      @export var Jump_Speed = 500
                      @export var Jump_Jetpack = true
                      @export var Gravity = 2500
                      @export var Gravity_Sphere = true
                      
                      func _ready():
                      	Input.set_mouse_mode(Input.MOUSE_MODE_CAPTURED)
                      
                      func _physics_process(delta):
                      	var input_dir = Input.get_vector(&"ui_left", &"ui_right", &"ui_up", &"ui_down")
                      	var accel = Accelaration if is_on_floor() else Air_Accelaration
                      	var decel = Decelaration if is_on_floor() else Air_Decelaration
                      
                      	# Apply velocity: Gravity, jump, movement, friction
                      	if not is_on_floor():
                      		var dist = 1 / (1 + global_position.distance_to(Vector3i(0, 0, 0)))
                      		velocity -= global_basis.y * abs(Gravity) * dist * delta
                      	if (Jump_Jetpack or is_on_floor()) and Input.is_action_pressed(&"ui_accept"):
                      		velocity += global_basis.y * abs(Jump_Speed) * delta
                      	velocity += (global_basis.x * input_dir.x + global_basis.z * input_dir.y) * accel * delta
                      	velocity /= 1 + decel * delta
                      	move_and_slide()
                      
                      	# Apply sphere orientation, overrides up_direction for move_and_slide to work correctly
                      	if Gravity_Sphere:
                      		var dir = +1 if Gravity >= 0 else -1
                      		global_basis.y = global_position.normalized() * dir
                      		global_basis.x = global_basis.y.cross(global_basis.z)
                      		global_basis = global_basis.orthonormalized()
                      		up_direction = global_basis.y
                      
                      func _input(event):
                      	if event is InputEventMouseMotion:
                      		rotate_object_local(Vector3.DOWN, event.relative.x * Sensitivity_X)
                      		$PlayerCamera.rotate_x(-event.relative.y * Sensitivity_Y)
                      		$PlayerCamera.rotation.x = min(deg_to_rad(Maximum_Y_Look), max(deg_to_rad(Minimum_Y_Look), $PlayerCamera.rotation.x))

                      There is but one issue left, and it would be amazing if anyone could give me a hand with it as well. As can be seen in the image, the atmosphere doesn't rotate with the player when using a PhysicalSkyMaterial, which causes the horizon to tilt and doesn't look realistic from behind: The horizon should always be down from the perspective of the player, the sun should shine from the same direction but moved to the new position on the rotated sky. If I create a script for my WorldEnvironment node to rotate it and its DirectionalLight3D child, what should it contain to get a proper results? It should probably have an offset for specifying the time of day, I could plug the time into it to get a daytime cycle and run it at a different phase for the moon.

                      • xyz replied to this.

                        Rather than transforming the character maybe transform the 'world'(planet)?

                        MirceaKitsune
                        Not sure I understand your environment problem. If you're walking on the surface of a spherical planet the environment could in reality get "tilted" in all kinds of ways. What's the meaning of having a "horizon" in planet's environment anyway? There should only be sun. The actual horizon is planet's contour.

                        But if you want the environment horizon to align with planet's horizon you'll need to use a custom sky/sun shader. Or do what @Megalomaniak said.

                          MirceaKitsune I managed to simplify it a bit so this is the only transform required for the base math:

                          global_basis.y = global_position.normalized()
                          global_basis.x = global_basis.y.cross(global_basis.z)
                          global_basis = global_basis.orthonormalized()

                          For a bulletproof solution, you need the second cross product as well. The third one in my example was redundant.
                          The order of cross products the engine does in Basis::orthonormalized() is not guaranteed. So if your code does not ensure orthogonality the whole thing may potentially misbehave.