Hello! I am having an issue with generating curves for my level generator. I must admit that math, mainly trig, is not my strong point, so I have kinda thrown together some kind of curve generator for use in my level creator.
Unfortunately, it doesn’t quite work. The curve is off-center from the start point from what appears to be how the math is set up (I added start and end points after, probably not the best practice) and the current curve is not a perfect 90 degree curve, meaning that the little “dot” (acorn) indicators at the start and end aren’t perpendicular.
If someone who is a lot more experienced that I am can help, or if there’s an easier way to do so (please note that the goal of this is to have a list of end points so that I can create code-based geometry with them) I would really appreciate it!

Below is my code and a rough diagram I made prior to trying to implement it with how I figured it'd work.

var points = [] # This will store the original points
var offset_distance = 1
var radius = 3

func _ready():
	# Get points in curve.
	var point
	# Append start point
	var start = Vector2(0, 0)
	points.append(start)
	
	# Calculate the points based on angles from 0 to 90 degrees
	var p_amt = 6
	for i in range(p_amt):
		# Find the current point.
		# Calculate angle in radians
		var angle = deg2rad(i * 90 / (p_amt - 1))
		# Scale to a radius (e.g., 2)
		point = Vector2(start.x - radius, start.y) + Vector2(cos(angle), sin(angle)) * radius
		points.append(point)
	
	# Append end point
	var end = Vector2(cos(deg2rad(90)), sin(deg2rad(90))) * radius
	points.append(end)
	
	for p in range(len(points) - 1):
		# Get point and next point.
		point = points[p]
		var next_point = points[p + 1]
		
		# Create the placeholder sprite to display the curve.
		var sprite = Sprite3D.new()
		sprite.texture = load("res://spr_list_item_mat_6.png")
		sprite.billboard = SpatialMaterial.BILLBOARD_ENABLED
		sprite.global_transform.origin = Vector3(point.x, 0, point.y)
		add_child(sprite)
		
		# Find the next point.
		var direction = (next_point - point).normalized()  # Get the normalized direction vector
		# Take that direction and distance and rotate it 90 degrees for the offset.
		var offset = Vector2(-direction.y, direction.x)
		
		# Turn new position into left and right points, offset by offset_distance.
		var left_point = point - offset * offset_distance
		var right_point = point + offset * offset_distance
		
		# Create left and right sprites at left and right points.
		var right_sprite = Sprite3D.new()
		right_sprite.texture = load("res://spr_list_item_mat_6.png")
		right_sprite.billboard = SpatialMaterial.BILLBOARD_ENABLED
		right_sprite.global_transform.origin = Vector3(right_point.x, 0, right_point.y)
		var left_sprite = Sprite3D.new()
		left_sprite.texture = load("res://spr_list_item_mat_6.png")
		left_sprite.billboard = SpatialMaterial.BILLBOARD_ENABLED
		left_sprite.global_transform.origin = Vector3(left_point.x, 0, left_point.y)
		
		# Add left and right placeholder sprites.
		add_child(left_sprite)
		add_child(right_sprite)

Also, as I forgot to include it, here is what currently happens:


Please note how the start and end are not purely perpendicular.

This may be helpful as a reference.

It's code I wrote to generate a torus-like 2D polygon.

extends Node2D

const OUTER_RADIUS: float = 50.0
const INNER_RADIUS: float = 34.0
const NUM_POINTS: int = 8
const SEGMENT_ANGLE: float = TAU / float(NUM_POINTS) # radians

func _ready() -> void:
	make_torus()

func make_torus() -> void:
	var angle: float # radians
	var points: PackedVector2Array = []

	# Start with a point on the torus' inner circle.
	angle = 0.0
	points.append(Vector2(cos(angle), sin(angle)) * INNER_RADIUS)

	# Move radially outward to the outer circle.
	# Move clockwise along the outer circle for the full circumference.
	for i in range(NUM_POINTS + 1):
		angle = float(i) * SEGMENT_ANGLE
		points.append(Vector2(cos(angle), sin(angle)) * OUTER_RADIUS)

	# Move radially inward to the inner circle.
	# Move counterclockwise along the inner circle for the full circumference.
	for i in range(NUM_POINTS):
		angle = -float(i) * SEGMENT_ANGLE
		points.append(Vector2(cos(angle), sin(angle)) * INNER_RADIUS)

	var polygon_2d: Polygon2D = Polygon2D.new()
	polygon_2d.polygon = points
	polygon_2d.color = Color.GOLDENROD
	add_child(polygon_2d)
~~`