User prompt
Please fix the bug: 'TypeError: obstacles[i].update is not a function' in or related to this line: 'obstacles[i].update();' Line Number: 529
User prompt
use a 2 dimension array for obstacle lines
Code edit (1 edits merged)
Please save this source code
User prompt
use the opponents array instead of oppenent1 in all the code where oppenent1 is used
User prompt
use the opponents array instead of oppenent1 in all the code where oppenent1 is used
User prompt
use the opponents array instead of oppenent1
Code edit (13 edits merged)
Please save this source code
User prompt
add a global variable for the number of opponents
User prompt
add an array for opponents
User prompt
add all request code for opponent2
User prompt
in the same way as opponnent1, add an opponnent2 with all necessery stuff
Code edit (1 edits merged)
Please save this source code
Code edit (1 edits merged)
Please save this source code
User prompt
in Athlete class use the line as an index in color array for the tint
Code edit (5 edits merged)
Please save this source code
User prompt
add an array of colors
Code edit (6 edits merged)
Please save this source code
User prompt
In athlete update() if isAi, call jump once when an obstacle of obstaclesLine1 is at a x distance of 300
Code edit (1 edits merged)
Please save this source code
Code edit (1 edits merged)
Please save this source code
Code edit (2 edits merged)
Please save this source code
User prompt
Please fix the bug: 'ReferenceError: opponent1 is not defined' in or related to this line: 'opponent1.update();' Line Number: 453
Code edit (5 edits merged)
Please save this source code
User prompt
Please fix the bug: 'Uncaught TypeError: Cannot read properties of undefined (reading 'rightArm')' in or related to this line: 'self.rightArm.rotation = idlePosture.rightArm.r;' Line Number: 236
User prompt
add an opponent1 athlete at x=400, y=groundLevel-200-220
/**** 
* Classes
****/ 
/****************************************************************************************** */ 
/**************************************** CLASSES ***************************************** */
/****************************************************************************************** */ 
/****************************************************************************************** */ 
/************************************* ATHLETE CLASS ************************************** */
/****************************************************************************************** */ 
var Athlete = Container.expand(function (line) {
	var self = Container.call(this);
	self.speedX = -globalSpeed;
	self.jumpSpeed = -17;
	self.gravity = 0.5;
	self.lineIndex = line;
	self.isAi = line > 0;
	self.isOnGround = true;
	self.isFalling = false; // Indicates if the athlete is currently falling
	self.currentPosture = ""; // Store the name of the current posture
	self.isChangingPosture = false; // 0 when idle or moving down, 1 when moving up
	self.nextPosture = null;
	self.tint = 0xc0d4FF;
	// BODY PARTS
	self.trunk = self.attachAsset('bodyPart', {
		anchorX: 0.5,
		anchorY: 0.5,
		scaleX: 1,
		scaleY: 1,
		rotation: 0.125,
		tint: 0x000000 //self.tint // TEMP DEBUG
	});
	self.head = self.trunk.attachAsset('head', {
		anchorX: 0.5,
		anchorY: 2,
		scaleX: 1,
		scaleY: 1,
		tint: 0x000000 // self.tint
	});
	self.rightArm = self.trunk.attachAsset('bodyPart', {
		anchorX: 0.5,
		anchorY: 0,
		scaleX: 1,
		scaleY: 1,
		tint: self.tint
	});
	// Right hand asset attachment removed
	self.leftArm = self.trunk.attachAsset('bodyPart', {
		anchorX: 0.5,
		anchorY: 0,
		scaleX: 1,
		scaleY: 1,
		tint: self.tint
	});
	// Left hand asset attachment removed
	self.rightLeg = self.trunk.attachAsset('bodyPart', {
		anchorX: 0.5,
		anchorY: 0,
		scaleX: 1,
		scaleY: 1.0,
		tint: self.tint
	});
	// Right foot asset attachment removed
	self.leftLeg = self.trunk.attachAsset('bodyPart', {
		anchorX: 0.5,
		anchorY: 0,
		scaleX: 1,
		scaleY: 1.0,
		tint: self.tint
	});
	// Left foot asset attachment removed
	// FUNCTIONS
	self.updatePosture = function () {
		var speed = 0.2; // Speed of transition
		if (!self.targetPosture) {
			log("no targetPosture.");
			return;
		}
		// Check if the target posture for hands and legs is reached
		//log("self.targetPosture: " + self.targetPosture.name + " :  self.rightHand.x: " + self.rightHand.x, " self.targetPosture.rightHand.x: " + self.targetPosture.rightHand.x);
		var handsAndLegsReached = Math.abs(self.rightLeg.x - self.targetPosture.rightLeg.x) < 1 && Math.abs(self.rightLeg.y - self.targetPosture.rightLeg.y) < 1 && Math.abs(self.leftLeg.x - self.targetPosture.leftLeg.x) < 1 && Math.abs(self.leftLeg.y - self.targetPosture.leftLeg.y) < 1 && Math.abs(self.head.x - self.targetPosture.head.x) < 1 && Math.abs(self.head.y - self.targetPosture.head.y) < 1;
		if (handsAndLegsReached) {
			log("handsAndLegsReached !");
			// Update currentPosture to the name of the targetPosture when hands and legs posture is reached
			self.currentPosture = self.targetPosture.name;
			self.targetPosture = null;
			self.isChangingPosture = false;
			if (self.nextPosture) {
				self.setPosture(self.nextPosture);
			}
			return;
		} else {
			self.isChangingPosture = true;
		}
		if (self.targetPosture && self.targetPosture.head) {
			self.head.x += (self.targetPosture.head.x - self.head.x) * speed;
		}
		self.trunk.rotation += (self.targetPosture.trunk.r - self.trunk.rotation) * speed;
		self.head.y += (self.targetPosture.head.y - self.head.y) * speed;
		self.rightArm.x += (self.targetPosture.rightArm.x - self.rightArm.x) * speed;
		self.rightArm.y += (self.targetPosture.rightArm.y - self.rightArm.y) * speed;
		self.rightArm.rotation += (self.targetPosture.rightArm.r - self.rightArm.rotation) * speed;
		log("Posture rightArm.rotation=" + self.rightArm.rotation);
		// Right hand updates removed due to hand assets being removed
		self.leftArm.x += (self.targetPosture.leftArm.x - self.leftArm.x) * speed;
		self.leftArm.y += (self.targetPosture.leftArm.y - self.leftArm.y) * speed;
		self.leftArm.rotation += (self.targetPosture.leftArm.r - self.leftArm.rotation) * speed;
		// Left hand updates removed due to hand assets being removed
		self.trunk.x += (self.targetPosture.trunk.x - self.trunk.x) * speed;
		self.trunk.y += (self.targetPosture.trunk.y - self.trunk.y) * speed;
		self.rightLeg.x += (self.targetPosture.rightLeg.x - self.rightLeg.x) * speed;
		self.rightLeg.y += (self.targetPosture.rightLeg.y - self.rightLeg.y) * speed;
		// Right foot updates removed due to foot assets being removed
		self.leftLeg.x += (self.targetPosture.leftLeg.x - self.leftLeg.x) * speed;
		self.leftLeg.y += (self.targetPosture.leftLeg.y - self.leftLeg.y) * speed;
	};
	self.moveRight = function () {
		self.x += self.speedX;
	};
	self.jump = function () {
		if (self.isOnGround) {
			self.isOnGround = false;
			self.speedY = self.jumpSpeed;
			log("--------------- JUMP --------------------");
		}
	};
	self.update = function () {
		if (!self.isFalling) {
			if (self.isOnGround) {
				self.leftLeg.height = 200;
				self.leftArm.height = 200;
				self.trunk.rotation = 0.125;
				self.runAnim();
			} else {
				self.speedY += self.gravity * Math.abs(globalSpeed / 10);
				self.y += self.speedY * Math.abs(globalSpeed / 10);
				log("self.speedY=" + self.speedY, " / self.y=" + self.y);
				if (self.y >= linesGroundLevels[self.lineIndex]) {
					self.y = linesGroundLevels[self.lineIndex];
					self.isOnGround = true;
					self.speedY = 0;
				} else {
					self.jumpAnim();
				}
			}
		} else {
			self.y += 5;
			if (self.y >= linesGroundLevels[self.lineIndex] + 200) {
				self.y = linesGroundLevels[self.lineIndex] + 200;
				self.isOnGround = true;
				self.speedY = 0;
			}
		}
		self.updatePosture();
		if (self.isAi) {
			if (athlete.isFalling) {
				self.moveRight();
			} else {
				self.x += 1 - Math.random();
			}
		}
	};
	self.runAnim = function () {
		var ocsilDelay = 5;
		var armsAmplitude = 0.5;
		var legsAmplitude = 0.5;
		// Simulate running by continuously balancing the arms and legs
		var runningArmAngle = Math.sin(LK.ticks / ocsilDelay) * armsAmplitude; // Oscillate arm angle to simulate running faster
		var runningLegAngle = Math.sin(LK.ticks / ocsilDelay) * legsAmplitude; // Oscillate leg angle to simulate running faster
		var armSwitchProgress = (Math.sin(LK.ticks / 5) + 0) / 2; // Normalize between 0 and 1
		self.rightArm.x = 10 * (1 - armSwitchProgress) - 5 * armSwitchProgress;
		self.leftArm.x = -15 * (1 - armSwitchProgress) + 20 * armSwitchProgress;
		self.rightLeg.x = 5 * (1 - armSwitchProgress) - 5 * armSwitchProgress;
		self.leftLeg.x = -10 * (1 - armSwitchProgress) + 10 * armSwitchProgress;
		log("Run rightArm.rotation=" + runningArmAngle);
		self.rightArm.rotation = runningArmAngle;
		self.leftArm.rotation = -runningArmAngle;
		self.rightLeg.rotation = -runningLegAngle;
		self.leftLeg.rotation = runningLegAngle;
		self.trunk.width = 30 - 10 * (runningArmAngle * Math.PI * 0.5);
		self.head.width = 50 - 10 * (runningArmAngle * Math.PI * 0.5);
	};
	self.jumpAnim = function () {
		var ocsilDelay = 50;
		var armsAmplitude = 1;
		// Simulate running by continuously balancing the arms and legs
		var runningArmAngle = Math.sin(LK.ticks / ocsilDelay) * armsAmplitude; // Oscillate arm angle to simulate running faster
		var armSwitchProgress = (Math.sin(LK.ticks / 5) + 0) / 2; // Normalize between 0 and 1
		self.rightArm.x = 10 * (1 - armSwitchProgress) - 5 * armSwitchProgress;
		self.leftArm.x = -15 * (1 - armSwitchProgress) + 20 * armSwitchProgress;
		self.rightLeg.x = 5 * (1 - armSwitchProgress) - 5 * armSwitchProgress;
		self.leftLeg.x = -10 * (1 - armSwitchProgress) + 10 * armSwitchProgress;
		self.trunk.width = 30 - 10 * (runningArmAngle * Math.PI * 0.5); //armSwitchProgress; // + 10 * Math.abs(runningArmAngle);
		self.trunk.rotation = 0.333; //armSwitchProgress; // + 10 * Math.abs(runningArmAngle);
		var rotationSpeed = 0.05; // Speed of rotation change
		if (self.speedY > 0) {
			self.rightArm.rotation += (0 - self.rightArm.rotation) * rotationSpeed % (Math.PI * 2);
			self.leftArm.rotation += (0 - self.leftArm.rotation) * rotationSpeed % (Math.PI * 2);
			// Return legs to initial rotation when descending
			self.rightLeg.rotation += (0 - self.rightLeg.rotation) * rotationSpeed;
			self.leftLeg.rotation += (0 - self.leftLeg.rotation) * rotationSpeed;
			self.leftLeg.height -= 30 * rotationSpeed;
			self.leftArm.height -= 20 * rotationSpeed;
		} else {
			log("self.rightArm.rotation=" + self.rightArm.rotation, " + " + -Math.PI * 3 * rotationSpeed % (Math.PI * 2));
			self.rightArm.rotation += -Math.PI * 3 * rotationSpeed % (Math.PI * 2);
			self.leftArm.rotation += Math.PI * 1.75 * rotationSpeed % (Math.PI * 2);
			// Continue with jump animation adjustments
			self.rightLeg.rotation += (-2.6 - self.rightLeg.rotation) * rotationSpeed;
			self.leftLeg.rotation += (1 - self.leftLeg.rotation) * rotationSpeed;
			self.leftLeg.height -= 0 * rotationSpeed;
			self.leftArm.height -= 35 * rotationSpeed;
		}
	};
	self.fallAnim = function () {
		var fallSpeed = 0.1; // Speed of fall animation
		// Adjust body parts to simulate falling
		self.trunk.rotation += (Math.PI / 2 - self.trunk.rotation) * fallSpeed;
		self.rightArm.rotation += (Math.PI / 2 - self.rightArm.rotation) * fallSpeed;
		self.leftArm.rotation += (Math.PI / 2 - self.leftArm.rotation) * fallSpeed;
		self.rightLeg.rotation += (Math.PI / 2 - self.rightLeg.rotation) * fallSpeed;
		self.leftLeg.rotation += (Math.PI / 2 - self.leftLeg.rotation) * fallSpeed;
	};
	self.setPosture = function (newPosture) {
		log("Set posture: " + newPosture.name);
		self.targetPosture = newPosture;
	};
	self.restore = function () {
		log("restore...");
		self.x = 400;
		self.y = linesGroundLevels[self.lineIndex];
		self.speedY = 0;
		self.trunk.rotation = 0;
		self.rightArm.rotation = idlePosture.rightArm.r;
		self.leftArm.rotation = 0;
		self.rightLeg.rotation = 0;
		self.leftLeg.rotation = 0;
		self.setPosture(idlePosture);
		self.isFalling = false;
		self.isOnGround = true;
		self.hasFlashed = false;
		self.hasFlashedGround = false;
	};
});
/****************************************************************************************** */ 
/************************************** OBSTACLE CLASS ************************************ */
/****************************************************************************************** */ 
// Obstacle class
var Obstacle = Container.expand(function () {
	var self = Container.call(this);
	self.leftRod = self.attachAsset('obstacle', {
		anchorX: 0.5,
		anchorY: 1.0 // Anchor at the bottom for collision detection
	});
	self.centralRod = self.attachAsset('obstacle', {
		anchorX: 0.5,
		anchorY: 1,
		y: -200,
		rotation: Math.PI * 0.125,
		tint: 0xffff00 // Apply a yellow tint for glow effect
	});
	self.rightRod = self.attachAsset('obstacle', {
		anchorX: 0.5,
		anchorY: 0.0,
		x: 95,
		y: -460
	});
	self.speedX = globalSpeed;
	self.update = function () {
		self.speedX = globalSpeed;
		self.x += self.speedX;
	};
});
/****************************************************************************************** */ 
/************************************** TRACK CLASS ************************************ */
/****************************************************************************************** */ 
var Track = Container.expand(function () {
	var self = Container.call(this);
	self.speedX = globalSpeed; // Use global speed for track movement
	// Attach track asset
	self.trackAsset = self.attachAsset('track', {
		anchorX: 0.0,
		anchorY: 0.0
	});
	// Method to update track position
	self.update = function () {
		self.speedX = globalSpeed;
		self.x += self.speedX;
		// Reset position to create a continuous track effect
		if (self.x <= -2048 - self.speedX * 2) {
			self.x = 2048 + self.speedX * 2;
		}
	};
});
/**** 
* Initialize Game
****/ 
var game = new LK.Game({
	backgroundColor: 0x87CEEB // Light blue background to represent the sky
});
/**** 
* Game Code
****/ 
/****************************************************************************************** */ 
/************************************** GLOBAL VARIABLES ********************************** */
/****************************************************************************************** */ 
// Enumeration for game states
var GAME_STATE = {
	INIT: 'INIT',
	MENU: 'MENU',
	HELP: 'HELP',
	STARTING: 'STARTING',
	NEW_ROUND: 'NEW_ROUND',
	PLAYING: 'PLAYING',
	SCORE: 'SCORE'
};
var gameState = GAME_STATE.INIT;
var athlete;
var opponent1;
var obstacles = [];
var obstaclesLine2 = [];
var groundLevel = 2732 - 200; // Ground level set 200px from the bottom
var linesGroundLevels = [groundLevel - 200, groundLevel - 420]; // Ground level set 200px from the bottom
var groundLevelLine1 = 2732 - 200; // Ground level set 200px from the bottom
var obstacleSpawnTicker = 0;
var obstacleSpawnRate = 180; // Spawn an obstacle every 2 seconds
var obstacleJustPassed = false;
var track;
var track2;
var track3;
var track4;
var scoreTxt;
var globalSpeed = -10; //-10; // Global speed for tracks and obstacles min = -4
var globalSpeedIncreaseStep = 5;
var isDebug = true;
var debugMarker;
/****************************************************************************************** */ 
/************************************** POSTURES ****************************************** */
/****************************************************************************************** */ 
var idlePosture = {
	name: "idlePosture",
	trunk: {
		x: 0,
		y: 0,
		r: 0.125
	},
	head: {
		x: 0,
		y: 0
	},
	rightArm: {
		x: 20,
		y: -80,
		r: 0 //Math.PI * 2 * 0.0
	},
	leftArm: {
		x: -20,
		y: -80,
		r: 0 //Math.PI * 2 * 0.125
	},
	rightLeg: {
		x: 5,
		y: 100
	},
	leftLeg: {
		x: -5,
		y: 100
	}
};
/****************************************************************************************** */ 
/*********************************** UTILITY FUNCTIONS ************************************ */
/****************************************************************************************** */ 
function log() {
	if (isDebug) {
		var _console;
		(_console = console).log.apply(_console, arguments);
	}
}
/****************************************************************************************** */ 
/************************************** INPUT HANDLERS ************************************ */
/****************************************************************************************** */ 
// Touch event to make the athlete jump
game.on('down', function () {
	athlete.jump();
	opponent1.jump();
});
/****************************************************************************************** */ 
/************************************* AI FUNCTIONS *************************************** */
/****************************************************************************************** */ 
/****************************************************************************************** */ 
/************************************* GAME STATES **************************************** */
/****************************************************************************************** */ 
function gameInitialize() {
	log("Game initialize...");
	// Add background asset
	scoreTxt = new Text2('0', {
		size: 150,
		fill: "#ffffff" // White color for better visibility
	});
	scoreTxt.anchor.set(0.5, 0); // Center the score text horizontally
	LK.gui.top.addChild(scoreTxt); // Add the score text to the GUI overlay at the top center
	var background = LK.getAsset('background', {
		anchorX: 0.0,
		anchorY: 0.0,
		visible: false // TEMP DEBUG
	});
	game.addChild(background);
	// Initialize track using Track class
	track = game.addChild(new Track());
	track.x = 0;
	track.y = 2732 - 512;
	// Second track instance for continuous effect
	track2 = game.addChild(new Track());
	track2.x = 2048; // Position the second track right after the first one
	track2.y = 2732 - 512;
	// Second line track 1
	track3 = game.addChild(new Track());
	track3.x = 0;
	track3.y = 2732 - 1024;
	// Second line track 2
	track4 = game.addChild(new Track());
	track4.x = 2048;
	track4.y = 2732 - 1024;
	// Initialize athlete
	athlete = game.addChild(new Athlete(0));
	athlete.restore();
	// Initialize opponents
	opponent1 = game.addChild(new Athlete(1));
	opponent1.restore();
	gameState = GAME_STATE.PLAYING;
	if (isDebug) {
		// Debug Marker
		debugMarker = LK.getAsset('debugMarker', {
			anchorX: 0.5,
			anchorY: 0.5
		});
		game.addChild(debugMarker);
		debugMarker.x = 0;
		debugMarker.y = groundLevel;
	}
}
function gamePlaying() {
	//log("Game playing...");
	// Spawn obstacles
	spawnObstacles();
	// Update obstacles
	updateObstacles();
	// Update tracks
	updateTracks();
	// Update Progress
	updateProgress();
	// Update athlete
	athlete.update();
	game.addChild(athlete);
	// Update opponents
	opponent1.update();
	game.addChild(opponent1);
	// Check for collisions
	checkForCollisions();
}
/****************************************************************************************** */ 
/************************************** GAME FUNCTIONS ************************************ */
/****************************************************************************************** */ 
function updateTracks() {
	if (!(athlete.isFalling && athlete.isOnGround)) {
		track.update();
		track2.update();
		track3.update();
		track4.update();
	}
}
function spawnObstacles() {
	// Spawn obstacles
	obstacleSpawnTicker += athlete.isFalling ? 0 : 1;
	if (obstacleSpawnTicker >= obstacleSpawnRate) {
		obstacleSpawnTicker = 0;
		// Line 1
		var newObstacle = new Obstacle(obstacles.length);
		newObstacle.x = 2048; // Start from the right edge
		newObstacle.y = groundLevel + 200;
		obstacles.push(newObstacle);
		game.addChild(newObstacle);
		// Line 2
		var newObstacle2 = new Obstacle(obstaclesLine2.length);
		newObstacle2.x = 2048 + 100; // Start from the right edge
		newObstacle2.y = groundLevel + 200 - 220;
		obstaclesLine2.push(newObstacle2);
		game.addChild(newObstacle2);
	}
}
function updateObstacles() {
	// Update obstacles
	for (var i = obstacles.length - 1; i >= 0; i--) {
		obstacles[i].update();
		// Remove obstacle if it moves off-screen and increase global speed
		if (obstacles[i].x < -100) {
			// Assuming obstacle width is less than 100px
			obstacles[i].destroy();
			obstacles.splice(i, 1);
			obstacleJustPassed = true;
		}
	}
	for (var i = obstaclesLine2.length - 1; i >= 0; i--) {
		obstaclesLine2[i].update();
		// Remove obstacle if it moves off-screen and increase global speed
		if (obstaclesLine2[i].x < -100) {
			// Assuming obstacle width is less than 100px
			obstaclesLine2[i].destroy();
			obstaclesLine2.splice(i, 1);
		}
	}
}
function updateProgress() {
	if (obstacleJustPassed) {
		// Update Progress
		obstacleJustPassed = false;
		LK.setScore(LK.getScore() + 1); // Add 1 to score for each hurdle passed only if not falling
		scoreTxt.setText(LK.getScore()); // Update score text with prefix
		// Increase global speed after each obstacle passed and add 1 to score only if not falling
		globalSpeed -= globalSpeedIncreaseStep; // Increase the speed
	}
}
function checkForCollisions() {
	obstacles.forEach(function (obstacle) {
		if (athlete.intersects(obstacle.centralRod)) {
			athlete.isFalling = true;
			athlete.isOnGround = false;
			athlete.fallAnim(); // Call fall animation before game over
		}
		if (opponent1.intersects(obstacle.centralRod)) {
			opponent1.isFalling = true;
			opponent1.isOnGround = false;
			opponent1.fallAnim(); // Call fall animation before game over
		}
	});
	handleFallEvent();
}
function handleFallEvent() {
	if (athlete.isFalling && !athlete.hasFlashed) {
		LK.effects.flashScreen(0xaaaaaa, 500); // Flash screen red for half a second
		athlete.hasFlashed = true;
		globalSpeed /= 1.5;
	}
	if (athlete.hasFlashed && !athlete.hasFlashedGround && athlete.isFalling && athlete.isOnGround) {
		LK.effects.flashScreen(0xff0000, 500); // Flash screen red for half a second
		athlete.hasFlashedGround = true;
		globalSpeed = 0;
		LK.setTimeout(function () {
			//LK.showGameOver(); // Show game over screen
			globalSpeed = -10;
			athlete.restore();
		}, 1000); // Delay game over to allow fall animation to be seen
	}
}
/****************************************************************************************** */ 
/************************************** MAIN GAME LOOP ************************************ */
/****************************************************************************************** */ 
LK.on('tick', function () {
	switch (gameState) {
		case GAME_STATE.MENU:
			// Handle menu logic here
			break;
		case GAME_STATE.STARTING:
			// Handle game starting logic here
			break;
		case GAME_STATE.PLAYING:
			gamePlaying();
			break;
		case GAME_STATE.SCORE:
			// Handle score display logic here
			break;
	}
});
gameInitialize(); ===================================================================
--- original.js
+++ change.js
@@ -6,13 +6,15 @@
 /****************************************************************************************** */ 
 /****************************************************************************************** */ 
 /************************************* ATHLETE CLASS ************************************** */
 /****************************************************************************************** */ 
-var Athlete = Container.expand(function () {
+var Athlete = Container.expand(function (line) {
 	var self = Container.call(this);
-	self.speedX = 5;
+	self.speedX = -globalSpeed;
 	self.jumpSpeed = -17;
 	self.gravity = 0.5;
+	self.lineIndex = line;
+	self.isAi = line > 0;
 	self.isOnGround = true;
 	self.isFalling = false; // Indicates if the athlete is currently falling
 	self.currentPosture = ""; // Store the name of the current posture
 	self.isChangingPosture = false; // 0 when idle or moving down, 1 when moving up
@@ -64,17 +66,8 @@
 		scaleX: 1,
 		scaleY: 1.0,
 		tint: self.tint
 	});
-	self.shadow = self.trunk.attachAsset('head', {
-		anchorX: 0,
-		anchorY: 1,
-		alpha: 0.5,
-		rotation: Math.PI * 0.5,
-		scaleX: 1,
-		scaleY: 1,
-		tint: 0x000000
-	});
 	// Left foot asset attachment removed
 	// FUNCTIONS
 	self.updatePosture = function () {
 		var speed = 0.2; // Speed of transition
@@ -140,26 +133,32 @@
 			} else {
 				self.speedY += self.gravity * Math.abs(globalSpeed / 10);
 				self.y += self.speedY * Math.abs(globalSpeed / 10);
 				log("self.speedY=" + self.speedY, " / self.y=" + self.y);
-				var groundY = groundLevel - 200;
-				if (self.y >= groundY) {
-					self.y = groundY;
+				if (self.y >= linesGroundLevels[self.lineIndex]) {
+					self.y = linesGroundLevels[self.lineIndex];
 					self.isOnGround = true;
 					self.speedY = 0;
 				} else {
 					self.jumpAnim();
 				}
 			}
 		} else {
 			self.y += 5;
-			if (self.y >= groundLevel) {
-				self.y = groundLevel;
+			if (self.y >= linesGroundLevels[self.lineIndex] + 200) {
+				self.y = linesGroundLevels[self.lineIndex] + 200;
 				self.isOnGround = true;
 				self.speedY = 0;
 			}
 		}
 		self.updatePosture();
+		if (self.isAi) {
+			if (athlete.isFalling) {
+				self.moveRight();
+			} else {
+				self.x += 1 - Math.random();
+			}
+		}
 	};
 	self.runAnim = function () {
 		var ocsilDelay = 5;
 		var armsAmplitude = 0.5;
@@ -227,9 +226,9 @@
 	};
 	self.restore = function () {
 		log("restore...");
 		self.x = 400;
-		self.y = groundLevel - 200;
+		self.y = linesGroundLevels[self.lineIndex];
 		self.speedY = 0;
 		self.trunk.rotation = 0;
 		self.rightArm.rotation = idlePosture.rightArm.r;
 		self.leftArm.rotation = 0;
@@ -321,8 +320,10 @@
 var opponent1;
 var obstacles = [];
 var obstaclesLine2 = [];
 var groundLevel = 2732 - 200; // Ground level set 200px from the bottom
+var linesGroundLevels = [groundLevel - 200, groundLevel - 420]; // Ground level set 200px from the bottom
+var groundLevelLine1 = 2732 - 200; // Ground level set 200px from the bottom
 var obstacleSpawnTicker = 0;
 var obstacleSpawnRate = 180; // Spawn an obstacle every 2 seconds
 var obstacleJustPassed = false;
 var track;
@@ -421,14 +422,13 @@
 	track4 = game.addChild(new Track());
 	track4.x = 2048;
 	track4.y = 2732 - 1024;
 	// Initialize athlete
-	athlete = game.addChild(new Athlete());
+	athlete = game.addChild(new Athlete(0));
 	athlete.restore();
 	// Initialize opponents
-	opponent1 = game.addChild(new Athlete());
-	opponent1.x = 400;
-	opponent1.y = groundLevel - 200 - 220;
+	opponent1 = game.addChild(new Athlete(1));
+	opponent1.restore();
 	gameState = GAME_STATE.PLAYING;
 	if (isDebug) {
 		// Debug Marker
 		debugMarker = LK.getAsset('debugMarker', {
:quality(85)/https://cdn.frvr.ai/661848ad3c9c5fe61a4c3f8e.png%3F3) 
 Elongated elipse with black top half and white bottom half.
:quality(85)/https://cdn.frvr.ai/661bdc7a58043b89d4ba9e89.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/661bdd1d58043b89d4ba9e8e.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/661be25558043b89d4ba9f0d.png%3F3) 
 full close and front view of empty stands. retro gaming style
:quality(85)/https://cdn.frvr.ai/661f4f57266989b7f2ed94f1.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/661f526b266989b7f2ed954f.png%3F3) 
 delete
:quality(85)/https://cdn.frvr.ai/661f5287266989b7f2ed9554.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66226cbbf46305276d9e4de1.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/6622735cf46305276d9e4e30.png%3F3) 
 delete
:quality(85)/https://cdn.frvr.ai/66228a13eff7d73bc2d1d4a9.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66228a2eeff7d73bc2d1d4ad.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/662384d91d31755ac753860d.png%3F3) 
 Basquettes à ressort futuriste. vue de profile. Retro gaming style
:quality(85)/https://cdn.frvr.ai/6623cf0f1d31755ac75387a8.png%3F3) 
 a blue iron man style armor flying. Retro gaming style
:quality(85)/https://cdn.frvr.ai/6623cf581d31755ac75387b3.png%3F3) 
 a blue iron man style armor flying horizontally. Retro gaming style
:quality(85)/https://cdn.frvr.ai/6623db441d31755ac753884e.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/6623dde21d31755ac753889e.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/662493106b90388bb0195cd3.png%3F3) 
 round button with a big "up" arrow icon and a small line under it. UI
:quality(85)/https://cdn.frvr.ai/6625589e6b90388bb0195eb5.png%3F3) 
 A big black horizontal arrow pointing left with centred text 'YOU' in capital letters, painted on an orange floor.. horizontal and pointing left
:quality(85)/https://cdn.frvr.ai/6625f7dfaefb923f60ced44a.png%3F3) 
 remove
:quality(85)/https://cdn.frvr.ai/66280aa325092f9d66ecfd6b.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66289a7225092f9d66ecfe81.png%3F3) 
 gold athletics medal with ribbon. retro gaming style
:quality(85)/https://cdn.frvr.ai/66294f1a777dbb646cb323ba.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66294f97777dbb646cb323d0.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66295338777dbb646cb32447.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/6629533f777dbb646cb3244b.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/6629546e777dbb646cb32460.png%3F3) 
 :quality(85)/https://cdn.frvr.ai/66295506777dbb646cb32474.png%3F3) 
 a black oval with a crying smiley face.
:quality(85)/https://cdn.frvr.ai/66295a3a777dbb646cb324d9.png%3F3)