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.

                        xyz Thanks for clarifying: I added just the second cross product back in that case.

                        For the environment: Imagine the sun is shining from any direction on a planet with an atmosphere. If you stay in an area where the sun is above (noon) the sun light is white and the sky is cyan colored, if you move away (dusk) sunlight is more yellow and the sky becomes dull toward the horizon, move further away (sunset) the sun becomes red at an angle as the sky grows dark. The default sky assumes -Y as down and can't easily change to reflect the atmosphere seen from different angles as the sun shines from the same direction. Most likely I need a different type of sky, possibly a custom shader like you said... getting a tilt corrected horizon to work would be a very complex issue in any case.

                        Even so I don't know what algorithm I'd need: I'll need some way to convert the player's position on the planet into a 0 to 1 range... 1 is the player standing right between the center of the planet and the direction in which the sun shines, 0 is the player standing on the exact opposite side of the planet from the directional light's perspective. I should be able to do the sun in just one axis which will help simplify this a lot, but the sky tilt still needs to be corrected in 2D or 3D space which is more problematic.