User prompt
Make face tracking lost text more visible
User prompt
Please fix the bug: 'TypeError: LK.showPauseMenu is not a function. (In 'LK.showPauseMenu('Face tracking lost. Please adjust your position.', { size: 80, fill: 0xFF0000 })', 'LK.showPauseMenu' is undefined)' in or related to this line: 'LK.showPauseMenu('Face tracking lost. Please adjust your position.', {' Line Number: 405
User prompt
Please fix the bug: 'TypeError: LK.pauseGame is not a function. (In 'LK.pauseGame()', 'LK.pauseGame' is undefined)' in or related to this line: 'LK.pauseGame();' Line Number: 405
User prompt
Pause and show a message in the middle of the screen when face tracking stops working ↪💡 Consider importing and using the following plugins: @upit/facekit.v1
Code edit (1 edits merged)
Please save this source code
User prompt
Make road 20% transparent
User prompt
Increase sensitivity of steering
Code edit (1 edits merged)
Please save this source code
User prompt
Ensure truck moves to middle of the lane when changing lanes
User prompt
Divide by 2 when calculating lane position for truck
User prompt
Divide by 2 when calculating lane position
User prompt
Show lane width in gui
User prompt
Please fix the bug: 'undefined is not an object (evaluating 'document.addEventListener')' in or related to this line: 'document.addEventListener('DOMContentLoaded', function () {' Line Number: 202
User prompt
Ensure dom is fully loaded before setting game width
User prompt
Display game width in the gui
User prompt
Make game width window.innerWidth * window.devicePixelRatio
User prompt
Move game width constant above lane width constant
User prompt
Move line 202 under line 203
User prompt
Set lane width to a third of game width
User prompt
Verify game width is actually the width of the screen
User prompt
Remove - 2 from line 161
User prompt
Renumber the lanes as 1, 2, and 3. Update all associated calculations and defaults
User prompt
If lane index is zero, targetx = lanewidth
User prompt
When calculating lane position, do not subtract 1 if position is 0
User prompt
Ensure lane index is never less than 0
/**** * Plugins ****/ var tween = LK.import("@upit/tween.v1"); var storage = LK.import("@upit/storage.v1", { highScore: 0 }); var facekit = LK.import("@upit/facekit.v1"); /**** * Classes ****/ var RoadLine = Container.expand(function () { var self = Container.call(this); self.speed = 0; // Will be set when spawned var lineGraphics = self.attachAsset('roadLine', { anchorX: 0.5, anchorY: 0.5 }); self.update = function () { self.y += self.speed; // Reset position when off screen if (self.y > 2732 + lineGraphics.height) { self.y = -lineGraphics.height; } }; return self; }); var RoadObject = Container.expand(function () { var self = Container.call(this); self.type = 'obstacle'; // Default type self.speed = 0; // Will be set when spawned // Create a shadow underneath the object var shadow = null; self.init = function (objectType, initialY) { self.type = objectType; self.y = initialY; // Remove any previous children while (self.children.length > 0) { self.removeChildAt(0); } // Create appropriate graphics based on type if (objectType === 'obstacle') { shadow = self.attachAsset('shadow', { anchorX: 0.5, anchorY: 0.5, alpha: 0.3, y: 50 }); self.graphics = self.attachAsset('obstacle', { anchorX: 0.5, anchorY: 0.5 }); } else if (objectType === 'fuel') { shadow = self.attachAsset('shadow', { anchorX: 0.5, anchorY: 0.5, alpha: 0.3, scaleX: 0.7, scaleY: 0.7, y: 60 }); self.graphics = self.attachAsset('fuel', { anchorX: 0.5, anchorY: 0.5 }); } else if (objectType === 'speedBoost') { shadow = self.attachAsset('shadow', { anchorX: 0.5, anchorY: 0.5, alpha: 0.3, scaleX: 0.6, scaleY: 0.6, y: 40 }); self.graphics = self.attachAsset('speedBoost', { anchorX: 0.5, anchorY: 0.5 }); } // Add floating animation for collectibles if (objectType !== 'obstacle') { self.startFloatingAnimation(); } else { // Randomly rotate obstacles slightly self.graphics.rotation = (Math.random() - 0.5) * 0.5; } }; self.startFloatingAnimation = function () { tween(self.graphics, { y: -10 }, { duration: 800, easing: tween.easeInOut, onFinish: function onFinish() { tween(self.graphics, { y: 10 }, { duration: 800, easing: tween.easeInOut, onFinish: function onFinish() { if (self.parent) { // Check if still in scene self.startFloatingAnimation(); } } }); } }); }; self.update = function () { self.y += self.speed; }; return self; }); var Truck = Container.expand(function () { var self = Container.call(this); // Create shadow underneath the truck var shadow = self.attachAsset('shadow', { anchorX: 0.5, anchorY: 0.5, alpha: 0.3, y: 130 }); // Create main truck asset var truckGraphics = self.attachAsset('truck', { anchorX: 0.5, anchorY: 0.5 }); // Set initial lane (middle lane) self.lane = 1; self.targetX = 0; self.moving = false; self.invulnerable = false; // Move to a specific lane (1-left, 2-middle, 3-right) self.moveToLane = function (laneIndex) { if (laneIndex < 1) { laneIndex = 1; } if (self.lane === laneIndex || self.moving) { return; } self.lane = laneIndex; self.moving = true; // Calculate lane position self.targetX = laneIndex * LANE_WIDTH; // Animate truck move tween(self, { x: self.targetX }, { duration: 250, easing: tween.easeInOut, onFinish: function onFinish() { self.moving = false; } }); }; // Make truck temporarily invulnerable self.setInvulnerable = function (duration) { self.invulnerable = true; // Flash effect var flashCount = 0; var flashInterval = LK.setInterval(function () { truckGraphics.alpha = truckGraphics.alpha === 1 ? 0.5 : 1; flashCount++; if (flashCount >= 10) { LK.clearInterval(flashInterval); truckGraphics.alpha = 1; self.invulnerable = false; } }, duration / 10); }; return self; }); /**** * Initialize Game ****/ var game = new LK.Game({ backgroundColor: 0x555555 }); /**** * Game Code ****/ // Game constants if (typeof document !== 'undefined') { document.addEventListener('DOMContentLoaded', function () { var GAME_WIDTH = window.innerWidth * window.devicePixelRatio; }); } else { var GAME_WIDTH = 2048; // Default width if not in a browser environment } var LANE_WIDTH = GAME_WIDTH / 3; var GAME_HEIGHT = 2732; var INITIAL_SPEED = 10; var MAX_SPEED = 25; var ACCELERATION = 0.001; var BASE_SPAWN_INTERVAL = 1200; // milliseconds var MIN_SPAWN_INTERVAL = 600; // Game variables var gameSpeed = INITIAL_SPEED; var distance = 0; var spawnInterval = BASE_SPAWN_INTERVAL; var lastObjectSpawn = 0; var objectPool = []; var leftLaneLines = []; var rightLaneLines = []; var activeObjects = []; var gameStarted = false; var isGameOver = false; // Create road background var road = game.addChild(LK.getAsset('road', { anchorX: 0.5, anchorY: 0, x: GAME_WIDTH / 2, y: 0 })); // Create lane dividers function createRoadLines() { // Left lane divider var leftX = GAME_WIDTH / 2 - LANE_WIDTH / 2; for (var i = 0; i < 20; i++) { var leftLine = new RoadLine(); leftLine.x = leftX; leftLine.y = -150 + i * 300; leftLine.speed = gameSpeed; leftLaneLines.push(leftLine); game.addChild(leftLine); } // Right lane divider var rightX = GAME_WIDTH / 2 + LANE_WIDTH / 2; for (var j = 0; j < 20; j++) { var rightLine = new RoadLine(); rightLine.x = rightX; rightLine.y = -150 + j * 300; rightLine.speed = gameSpeed; rightLaneLines.push(rightLine); game.addChild(rightLine); } } // Create our truck var truck = new Truck(); truck.x = GAME_WIDTH / 2; // Middle lane truck.y = GAME_HEIGHT - 400; // Near bottom of screen // Add score display var scoreText = new Text2('Distance: 0m', { size: 70, fill: 0xFFFFFF }); scoreText.anchor.set(0.5, 0); LK.gui.top.addChild(scoreText); // Display game width var gameWidthText = new Text2('Width: ' + GAME_WIDTH + 'px', { size: 50, fill: 0xFFFFFF }); gameWidthText.anchor.set(0.5, 0); gameWidthText.y = 40; LK.gui.top.addChild(gameWidthText); // Display lane width var laneWidthText = new Text2('Lane Width: ' + LANE_WIDTH + 'px', { size: 50, fill: 0xFFFFFF }); laneWidthText.anchor.set(0.5, 0); laneWidthText.y = 60; LK.gui.top.addChild(laneWidthText); // Add high score display var highScoreText = new Text2('Best: 0m', { size: 50, fill: 0xFFFFFF }); highScoreText.anchor.set(0.5, 0); highScoreText.y = 80; LK.gui.top.addChild(highScoreText); // Face guidance text var faceGuideText = new Text2('Tilt your head to steer', { size: 60, fill: 0xFFFFFF }); faceGuideText.anchor.set(0.5, 0); faceGuideText.y = 150; LK.gui.top.addChild(faceGuideText); // Function to get a reusable object from the pool or create a new one function getObjectFromPool() { for (var i = 0; i < objectPool.length; i++) { if (!objectPool[i].parent) { return objectPool[i]; } } // Create a new object if none available var newObject = new RoadObject(); objectPool.push(newObject); return newObject; } // Function to spawn a random object on the road function spawnRoadObject() { // Determine object type (mostly obstacles, sometimes collectibles) var objectType = Math.random() < 0.7 ? 'obstacle' : Math.random() < 0.5 ? 'fuel' : 'speedBoost'; // Determine lane var lane = Math.floor(Math.random() * 3) + 1; // 1, 2, or 3 var laneX = (lane - 2) * LANE_WIDTH + GAME_WIDTH / 2; // Get object from pool and configure it var roadObject = getObjectFromPool(); roadObject.init(objectType, -200); // Start above the screen roadObject.x = laneX; roadObject.speed = gameSpeed; // Add to game and active objects list game.addChild(roadObject); activeObjects.push(roadObject); // Schedule next spawn lastObjectSpawn = LK.ticks; spawnInterval = Math.max(MIN_SPAWN_INTERVAL, BASE_SPAWN_INTERVAL - distance / 100); } // Initialize the high score function initHighScore() { highScoreText.setText('Best: ' + storage.highScore + 'm'); } // Start or restart the game function startGame() { // Reset game variables gameSpeed = INITIAL_SPEED; distance = 0; spawnInterval = BASE_SPAWN_INTERVAL; lastObjectSpawn = 0; isGameOver = false; // Clear any active objects while (activeObjects.length > 0) { var obj = activeObjects.pop(); if (obj.parent) { obj.parent.removeChild(obj); } } // Create road lines if not already created if (leftLaneLines.length === 0) { createRoadLines(); } // Reset truck position truck.lane = 2; truck.x = GAME_WIDTH / 2; truck.moveToLane(2); // Ensure truck starts in the center lane game.addChild(truck); // Update score display scoreText.setText('Distance: 0m'); initHighScore(); // Mark game as started gameStarted = true; // Play background music LK.playMusic('bgMusic', { fade: { start: 0, end: 0.3, duration: 1000 } }); } // Handle game over function gameOver() { if (isGameOver) { return; } isGameOver = true; LK.getSound('crash').play(); // Fade out music LK.playMusic('bgMusic', { fade: { start: 0.3, end: 0, duration: 800 } }); // Update high score if needed if (distance > storage.highScore) { storage.highScore = Math.floor(distance); highScoreText.setText('Best: ' + storage.highScore + 'm'); } // Show red flash effect LK.effects.flashScreen(0xFF0000, 1000); // Show game over screen LK.setTimeout(function () { LK.showGameOver(); }, 1000); } // Handle face tracking for truck steering function updateTruckPosition() { if (!gameStarted || isGameOver) { return; } // Get face position var headPosition = facekit.noseTip; if (headPosition && headPosition.x > 0) { // Normalize position and map to lanes var relativeX = headPosition.x / GAME_WIDTH; if (relativeX < 0.4) { truck.moveToLane(1); // Left lane } else if (relativeX > 0.6) { truck.moveToLane(3); // Right lane } else { truck.moveToLane(2); // Center lane } } else { // If no head position detected, move to center lane truck.moveToLane(1); } // Hide face guide text after player has moved if (faceGuideText.parent) { faceGuideText.parent.removeChild(faceGuideText); } } // Check collisions between truck and road objects function checkCollisions() { if (isGameOver) { return; } for (var i = activeObjects.length - 1; i >= 0; i--) { var obj = activeObjects[i]; // Check if object is on screen if (obj.y > GAME_HEIGHT + 200) { game.removeChild(obj); activeObjects.splice(i, 1); continue; } // Check for collision with truck if (truck.intersects(obj)) { if (obj.type === 'obstacle') { // Hit obstacle if (!truck.invulnerable) { gameOver(); } } else if (obj.type === 'fuel') { // Collected fuel LK.getSound('collect').play(); LK.setScore(LK.getScore() + 50); truck.setInvulnerable(2000); } else if (obj.type === 'speedBoost') { // Speed boost LK.getSound('speedUp').play(); gameSpeed += 5; if (gameSpeed > MAX_SPEED) { gameSpeed = MAX_SPEED; } } // Remove collected/hit object game.removeChild(obj); activeObjects.splice(i, 1); } } } // Game start with face detection game.down = function (x, y, obj) { if (!gameStarted) { startGame(); } }; // Game update loop game.update = function () { if (!gameStarted) { return; } // Update game speed and distance gameSpeed = Math.min(MAX_SPEED, INITIAL_SPEED + distance * ACCELERATION); distance += gameSpeed / 60; // Update score display scoreText.setText('Distance: ' + Math.floor(distance) + 'm'); LK.setScore(Math.floor(distance)); // Update road lines speed for (var i = 0; i < leftLaneLines.length; i++) { leftLaneLines[i].speed = gameSpeed; rightLaneLines[i].speed = gameSpeed; } // Update all active objects' speed for (var j = 0; j < activeObjects.length; j++) { activeObjects[j].speed = gameSpeed; } // Handle face tracking for truck steering updateTruckPosition(); // Check if it's time to spawn a new object if (LK.ticks - lastObjectSpawn > spawnInterval / (1000 / 60)) { spawnRoadObject(); } // Check for collisions checkCollisions(); }; // Initialize high score on load initHighScore();
===================================================================
--- original.js
+++ change.js
@@ -257,8 +257,16 @@
});
gameWidthText.anchor.set(0.5, 0);
gameWidthText.y = 40;
LK.gui.top.addChild(gameWidthText);
+// Display lane width
+var laneWidthText = new Text2('Lane Width: ' + LANE_WIDTH + 'px', {
+ size: 50,
+ fill: 0xFFFFFF
+});
+laneWidthText.anchor.set(0.5, 0);
+laneWidthText.y = 60;
+LK.gui.top.addChild(laneWidthText);
// Add high score display
var highScoreText = new Text2('Best: 0m', {
size: 50,
fill: 0xFFFFFF