User prompt
Player can move their vehicle by touching anywhere on the screen. The vehicle moves relative to the speed and direction at which the player moves their finger
User prompt
Why isn’t the touch event handler working?
User prompt
The player can move their vehicle by dragging it with one finger
User prompt
The player vehicle continuously shoots bullets that move from the front of the vehicle to the top of the screen
User prompt
Double the sprite size
User prompt
Player vehicle should start at bottom of the screen
User prompt
Initialize player’s vehicle sprite
User prompt
Neon Drift: Sky Havoc
Initial prompt
Top down cyberpunk themed shoot em up where you pilot a flying car
/**** * Plugins ****/ var tween = LK.import("@upit/tween.v1"); var storage = LK.import("@upit/storage.v1", { highScore: 0 }); /**** * Classes ****/ // Bullet class var Bullet = Container.expand(function () { var self = Container.call(this); // Attach the bullet sprite var bulletSprite = self.attachAsset('bullet', { anchorX: 0.5, anchorY: 0.5 }); // Set bullet speed self.speed = -10 + Math.random() * 2 - 1; // Randomize speed slightly between -11 and -9 // This is automatically called every game tick, if the bullet is attached! self.update = function () { self.y += self.speed; // Destroy the bullet if it goes off the screen if (self.y < -self.height) { if (self.hitbox) { if (self.hitbox.hitboxGraphics) { self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when bullet is destroyed } self.hitbox.destroy(); // Ensure hitbox is removed when bullet is destroyed } self.destroy(); // Ensure bullet container is removed from the bullets array var index = bullets.indexOf(self); if (index > -1) { bullets.splice(index, 1); } } // Check for collision with enemies for (var i = 0; i < enemies.length; i++) { if (self.intersects(enemies[i]) && enemies[i].isFullyOnScreen()) { enemies[i].takeDamage(); if (self.hitbox) { if (self.hitbox.hitboxGraphics) { self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when bullet is destroyed upon collision } self.hitbox.destroy(); // Ensure hitbox is removed when bullet is destroyed upon collision } self.destroy(); break; } } }; }); // CityTiles class var CityTiles = Container.expand(function () { var self = Container.call(this); // Attach the city tiles sprite var cityTilesSprite = self.attachAsset('CityTiles', { anchorX: 0.5, anchorY: 0.5, scaleX: 4, scaleY: 4 }); // Set city tiles speed self.speed = 0.1; // This is automatically called every game tick, if the city tiles are attached! self.update = function () { self.y += self.speed; // Reset the position if the city tiles go off the screen if (self.y > 2732 + self.height / 2) { self.y = -self.height / 2; } }; }); // Cloud class var Cloud = Container.expand(function () { var self = Container.call(this); // Randomly assign cloud asset var cloudAssets = ['cloud1', 'cloud2', 'cloud3', 'cloud4', 'cloud5']; var randomCloud = cloudAssets.splice(Math.floor(Math.random() * cloudAssets.length), 1)[0]; var scale = Math.random() * 0.5 + 0.5; // Mirror cloud sprites 50% of the time var flipX = Math.random() > 0.5 ? -1 : 1; var cloudSprite = self.attachAsset(randomCloud, { anchorX: 0.5, anchorY: 0.5, scaleX: scale * flipX, scaleY: scale }); // Set cloud sprite to be 70% transparent cloudSprite.alpha = 0.1; // Set cloud speed self.speed = Math.random() * 3 + 2; // This is automatically called every game tick, if the cloud is attached! self.update = function () { self.y += self.speed * 2 * (1 + self.x / 2048 * 0.5); // Remove the cloud if it goes off the screen if (self.y > 2732 + self.height / 2) { // Remove the cloud from the clouds array var index = clouds.indexOf(self); if (index > -1) { clouds.splice(index, 1); } cloudAssets.push(randomCloud); // Add the cloud back to the cloudAssets array // Destroy the cloud self.destroy(); } }; }); // Enemy class var Enemy = Container.expand(function () { var self = Container.call(this); self.isFullyOnScreen = function () { return self.x - self.width / 2 >= 0 && self.x + self.width / 2 <= 2048 && self.y - self.height / 2 >= 0 && self.y + self.height / 2 <= 2732; }; // This is automatically called every game tick, if the enemy is attached! var enemyColors = [0xCC6666, 0x66CC66, 0x6666CC, 0xCCCC66, 0xCC66CC, 0x66CCCC, 0xCC9966, 0x99CC66, 0x6699CC, 0x9966CC]; // Define a more varied palette of colors var randomColor = enemyColors[Math.floor(Math.random() * enemyColors.length)]; // Randomly select a color var enemySprite = self.attachAsset('enemy', { anchorX: 0.5, anchorY: 0.5, tint: randomColor // Apply the random color as a tint }); // Create a hitbox for collision detection self.hitbox = new Container(); self.hitbox.width = enemySprite.width * 0.5; self.hitbox.height = enemySprite.height * 0.5; self.hitbox.x = -self.hitbox.width / 2; // Center the hitbox self.hitbox.y = -self.hitbox.height / 2; // Center the hitbox self.addChild(self.hitbox); // Draw the hitbox for visualization var hitboxGraphics = LK.getAsset('enemy', { width: self.hitbox.width, height: self.hitbox.height, color: 0xff0000, // Red color for hitbox shape: 'box' }); self.hitbox.hitboxGraphics = hitboxGraphics; self.hitbox.addChild(self.hitbox.hitboxGraphics); // Set enemy speed self.speed = 2; // Initialize hit points self.hitPoints = 3; // Method to handle enemy damage self.takeDamage = function () { self.hitPoints -= 1; // Flash enemy yellow when hit tween(self, { tint: 0xFFFF00 }, { duration: 100, easing: tween.linear, onFinish: function onFinish() { tween(self, { tint: randomColor // Return to original random color }, { duration: 100, easing: tween.linear }); } }); if (self.hitPoints <= 0) { if (self.hitbox) { self.hitbox.destroy(); // Ensure hitbox is removed when enemy is destroyed } self.destroy(); score += 10; // Increase score by 10 for each enemy destroyed if (typeof scoreTxt !== 'undefined') { scoreTxt.setText(score.toString()); // Update score display if (score > highScore) { highScore = score; storage.highScore = highScore; // Persist new high score } } // Ensure the enemy container is removed from the enemies array var index = enemies.indexOf(self); if (index > -1) { enemies.splice(index, 1); } } }; // This is automatically called every game tick, if the enemy is attached! self.update = function () { self.y += self.speed; if (typeof self.speedX !== 'undefined') { self.x += self.speedX; // Move horizontally if speedX is defined if (self.x < 0 || self.x > 2048) { self.speedX *= -1; // Reverse direction if hitting screen edge } } // Check for overlap with other enemies and adjust position for (var i = 0; i < enemies.length; i++) { if (enemies[i] !== self) { var dx = self.x - enemies[i].x; var dy = self.y - enemies[i].y; var distance = Math.sqrt(dx * dx + dy * dy); if (distance < self.width) { if (dx < 0) { self.x -= 1; // Move left to avoid overlap } else { self.x += 1; // Move right to avoid overlap } } } } // Implement dodging behavior if (typeof playerVehicle !== 'undefined') { if (self.x < playerVehicle.x && self.x < 2048 - self.width / 2) { self.x += 1; // Move right to dodge } else if (self.x > playerVehicle.x && self.x > self.width / 2) { self.x -= 1; // Move left to dodge } } // Destroy the enemy if it goes off the screen if (self.y > 2732 + self.height) { self.destroy(); } // Fire a projectile every 180 ticks if (typeof self.lastTick === 'undefined') { self.lastTick = LK.ticks; } if (LK.ticks % 180 === 0) { var projectile = new Projectile(); projectile.x = self.x; projectile.y = self.y + self.height / 2 + projectile.height / 2; game.addChild(projectile); self.lastTick = LK.ticks; } }; }); // PlayerLifeBar class var PlayerLifeBar = Container.expand(function () { var self = Container.call(this); // Create a background bar var backgroundBar = self.attachAsset('playerHealthBackground', { anchorX: 0.5, anchorY: 0.5, scaleX: 1.0, scaleY: 1.0 }); backgroundBar.alpha = 0.5; // Set semi-transparency // Create an array to hold the life segments self.lifeSegments = []; for (var i = 0; i < 10; i++) { var lifeSegment = self.attachAsset('playerHealthBar', { anchorX: 0.5, anchorY: 0.5, scaleX: 1.0, // Keep original scale for life segments scaleY: 1.0, color: 0x00ff00, // Green for life alpha: 0.5 // Set semi-transparency }); lifeSegment.x = (i - 4.5) * 100; // Position segments from left to right with 100px width self.lifeSegments.push(lifeSegment); } // Method to update the life bar based on player's hit points self.updateLife = function (hitPoints) { for (var i = 0; i < self.lifeSegments.length; i++) { self.lifeSegments[i].visible = i < hitPoints; // Show segments based on hit points } }; }); // Player's vehicle class var PlayerVehicle = Container.expand(function () { var self = Container.call(this); // Attach the player's vehicle sprite var vehicleSprite = self.attachAsset('playerVehicle', { anchorX: 0.5, anchorY: 0.5, scaleX: 2, scaleY: 2 }); // This is automatically called every game tick, if the player's vehicle is attached! self.update = function () { // Fire a bullet every 30 ticks if (LK.ticks % 30 == 0) { var bullet = new Bullet(); bullet.x = self.x; bullet.y = self.y - self.height / 2 - bullet.height / 2; game.addChild(bullet); bullets.push(bullet); } }; // Add touch event handlers to the player's vehicle to enable dragging self.down = function (x, y, obj) { // Store the initial touch position self.initialTouchPosition = { x: x, y: y }; // Store the initial position of the player's vehicle self.initialPosition = { x: self.x, y: self.y }; }; self.move = function (x, y, obj) { // Update the position of the player's vehicle relative to the speed and direction at which the player moves their finger // Calculate the intended movement var intendedX = self.initialPosition.x + (x - self.initialTouchPosition.x); var intendedY = self.initialPosition.y + (y - self.initialTouchPosition.y); // Calculate the distance to move var deltaX = intendedX - self.x; var deltaY = intendedY - self.y; // Calculate the speed limit var maxSpeed = 10; // Maximum speed per frame // Calculate the actual movement based on the speed limit var distance = Math.sqrt(deltaX * deltaX + deltaY * deltaY); if (distance > maxSpeed) { var ratio = maxSpeed / distance; deltaX *= ratio; deltaY *= ratio; } // Update the position with the limited speed self.x += deltaX; self.y += deltaY; }; self.up = function (x, y, obj) { // Reset the initial touch position and the initial position of the player's vehicle self.initialTouchPosition = null; self.initialPosition = null; }; // Method to handle player damage self.hitPoints = 10; // Initialize player hit points self.takeDamage = function () { self.hitPoints -= 1; if (typeof playerLifeBar !== 'undefined') { playerLifeBar.updateLife(self.hitPoints); // Update the life bar if (self.hitPoints < playerLifeBar.lifeSegments.length) { playerLifeBar.lifeSegments[self.hitPoints].visible = false; // Hide the rightmost segment } } // Blink effect using tween tween(self, { alpha: 0.5 }, { duration: 100, easing: tween.linear, onFinish: function onFinish() { tween(self, { alpha: 1 }, { duration: 100, easing: tween.linear }); } }); if (self.hitPoints <= 0) { self.destroy(); // Destroy the player vehicle if hit points reach zero LK.showGameOver(); // Trigger game over when player health is depleted } }; }); // Projectile class var Projectile = Container.expand(function () { var self = Container.call(this); // Attach the projectile sprite var projectileSprite = self.attachAsset('projectile', { anchorX: 0.5, anchorY: 0.5, scaleX: 1.0, // Keep original scale for sprite scaleY: 1.0, // Keep original scale for sprite width: 100, // Set sprite width to 100 height: 100 // Set sprite height to 100 }); // Create a hitbox for collision detection var hitbox = new Container(); hitbox.width = projectileSprite.width * 0.5; hitbox.height = projectileSprite.height * 0.5; hitbox.x = -hitbox.width / 2; // Center the hitbox hitbox.y = -hitbox.height / 2; // Center the hitbox self.addChild(hitbox); // Draw the hitbox for visualization var hitboxGraphics = LK.getAsset('projectile', { width: hitbox.width, height: hitbox.height, color: 0xff0000, // Red color for hitbox shape: 'box' }); hitbox.addChild(hitboxGraphics); // Set projectile speed self.speed = 5; // This is automatically called every game tick, if the projectile is attached! self.update = function () { if (self.lastX === undefined) { self.lastX = self.x; } if (self.lastY === undefined) { self.lastY = self.y; } self.y += self.speed; self.lastX = self.x; self.lastY = self.y; // Destroy the projectile if it goes off the screen if (self.y > 2732 + self.height / 2) { self.destroy(); } // Check for collision with player // Ensure playerVehicle is defined before checking intersection if (typeof playerVehicle !== 'undefined' && self.intersects(playerVehicle.hitbox)) { // Damage the player playerVehicle.takeDamage(); // Destroy the projectile self.destroy(); if (self.hitbox) { self.hitbox.destroy(); // Ensure hitbox is removed when projectile is destroyed if (self.hitbox.hitboxGraphics) { self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when projectile is destroyed } } } // Check for collision with bullets for (var i = 0; i < bullets.length; i++) { var dx = self.x - bullets[i].x; var dy = self.y - bullets[i].y; var distance = Math.sqrt(dx * dx + dy * dy); if (distance < self.width / 2 + bullets[i].width / 2) { // Allow bullets to pass through projectiles without being destroyed continue; } } }; }); /**** * Initialize Game ****/ // Add player's vehicle to the game var game = new LK.Game({ backgroundColor: 0x000000 }); /**** * Game Code ****/ // Initialize score and display it at the top of the screen var score = 0; var highScore = storage.highScore || 0; var scoreTxt = new Text2(score.toString(), { size: 100, fill: 0xFFFFFF }); scoreTxt.anchor.set(0.5, 0); var highScoreTxt = new Text2("High Score: " + highScore.toString(), { size: 50, fill: 0xFFFFFF }); highScoreTxt.anchor.set(0.5, 0); highScoreTxt.y = scoreTxt.height; // Position below the score LK.gui.top.addChild(highScoreTxt); LK.gui.top.addChild(scoreTxt); // Create an array to store the bullets var bullets = []; // Create an array to store the enemies var enemies = []; // Create a timer to add enemies in waves var enemyWaveTimer = LK.setInterval(function () { // Add a wave of 3 enemies to reduce on-screen enemies for (var i = 0; i < 2; i++) { var enemy = game.addChild(new Enemy()); enemy.y = -enemy.height; enemy.x = (i + 1) * (2048 / 6); // Add the new enemy to the enemies array enemies.push(enemy); } // Add diagonal spawn pattern for (var i = 0; i < 2; i++) { var enemy = game.addChild(new Enemy()); enemy.y = -enemy.height - i * 100; // Staggered start positions enemy.x = i % 2 === 0 ? 0 : 2048; // Alternate starting from left and right enemy.speedX = i % 2 === 0 ? 1 : -1; // Move right if starting from left, left if starting from right enemies.push(enemy); } // Set a new random interval for the next wave LK.clearInterval(enemyWaveTimer); enemyWaveTimer = LK.setInterval(arguments.callee, Math.random() * 2000 + 4000); // Random interval between 4 and 6 seconds }, Math.random() * 2000 + 4000); // Initial random interval between 4 and 6 seconds // Increase enemy speed and spawn rate every time score reaches a multiple of 200 if (score % 200 === 0 && score !== 0) { enemies.forEach(function (enemy) { enemy.speed += 0.5; // Increase enemy speed }); LK.clearInterval(enemyWaveTimer); enemyWaveTimer = LK.setInterval(function () { for (var i = 0; i < 3; i++) { var enemy = game.addChild(new Enemy()); enemy.y = -enemy.height; enemy.x = (i + 1) * (2048 / 6); enemies.push(enemy); } // Add zigzag spawn pattern for (var i = 0; i < 2; i++) { var enemy = game.addChild(new Enemy()); enemy.y = -enemy.height; enemy.x = i % 2 === 0 ? 0 : 2048; // Start from left or right enemy.speedX = i % 2 === 0 ? 2 : -2; // Move in zigzag pattern enemies.push(enemy); } }, Math.max(1000, 5000 - score / 200 * 500)); // Decrease interval time } // Add CityTiles to the game and repeat to fill the screen horizontally var cityTilesArray = []; for (var i = 0; i < Math.ceil(2048 / 804) + 1; i++) { // 804 is the new width of a single CityTile for (var j = 0; j < Math.ceil(2732 / 804) + 1; j++) { // 804 is the new height of a single CityTile var cityTile = game.addChild(new CityTiles()); cityTile.x = i * 804 - 402; // Adjust position for seamless tiling cityTile.y = j * 804 - 402; // Adjust position for seamless tiling cityTilesArray.push(cityTile); } } // Add player's vehicle to the game var playerVehicle = game.addChild(new PlayerVehicle()); playerVehicle.hitbox = new Container(); playerVehicle.hitbox.width = playerVehicle.width * 0.5; // Define hitbox size playerVehicle.hitbox.height = playerVehicle.height * 0.5; // Define hitbox size playerVehicle.hitbox.x = -playerVehicle.hitbox.width / 2; // Center the hitbox playerVehicle.hitbox.y = -playerVehicle.hitbox.height / 2; // Center the hitbox playerVehicle.addChild(playerVehicle.hitbox); // Draw the hitbox for visualization var playerHitboxGraphics = LK.getAsset('playerHitbox', { width: playerVehicle.hitbox.width, height: playerVehicle.hitbox.height, color: 0x00ff00, // Green color for hitbox shape: 'box' }); playerVehicle.hitbox.addChild(playerHitboxGraphics); playerVehicle.x = 2048 / 2; playerVehicle.y = 2732 - playerVehicle.height / 2; // Add player's life bar to the game var playerLifeBar = game.addChild(new PlayerLifeBar()); playerLifeBar.x = playerLifeBar.width / 2 + 10; // Position the life bar at the lower left corner with a 10 pixel margin playerLifeBar.y = 2732 - playerLifeBar.height / 2 - 10; // Adjust y position to be at the bottom with a 10 pixel margin var dragNode = null; game.down = function (x, y, obj) { dragNode = playerVehicle; dragNode.initialTouchPosition = { x: x, y: y }; dragNode.initialPosition = { x: playerVehicle.x, y: playerVehicle.y }; }; game.move = function (x, y, obj) { if (dragNode && typeof playerVehicle !== 'undefined') { var newX = dragNode.initialPosition.x + (x - dragNode.initialTouchPosition.x); var newY = dragNode.initialPosition.y + (y - dragNode.initialTouchPosition.y); // Limit player vehicle movement to within the screen newX = Math.max(Math.min(newX, 2048 - dragNode.width / 2), dragNode.width / 2); newY = Math.max(Math.min(newY, 2732 - dragNode.height / 2), dragNode.height / 2); dragNode.x = newX; dragNode.y = newY; } }; game.up = function (x, y, obj) { if (dragNode) { dragNode = null; } }; // Create an array to store the clouds var clouds = []; // Create a timer to add clouds at random intervals between 1 and 5 seconds var cloudTimer = LK.setInterval(function () { // Check if there are less than 4 clouds on screen if (clouds.length < 4) { var cloud = game.addChild(new Cloud()); cloud.y = -cloud.height; cloud.x = Math.random() * 2048; // Add the new cloud to the clouds array clouds.push(cloud); } }, Math.random() * 4000 + 1000);
/****
* Plugins
****/
var tween = LK.import("@upit/tween.v1");
var storage = LK.import("@upit/storage.v1", {
highScore: 0
});
/****
* Classes
****/
// Bullet class
var Bullet = Container.expand(function () {
var self = Container.call(this);
// Attach the bullet sprite
var bulletSprite = self.attachAsset('bullet', {
anchorX: 0.5,
anchorY: 0.5
});
// Set bullet speed
self.speed = -10 + Math.random() * 2 - 1; // Randomize speed slightly between -11 and -9
// This is automatically called every game tick, if the bullet is attached!
self.update = function () {
self.y += self.speed;
// Destroy the bullet if it goes off the screen
if (self.y < -self.height) {
if (self.hitbox) {
if (self.hitbox.hitboxGraphics) {
self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when bullet is destroyed
}
self.hitbox.destroy(); // Ensure hitbox is removed when bullet is destroyed
}
self.destroy();
// Ensure bullet container is removed from the bullets array
var index = bullets.indexOf(self);
if (index > -1) {
bullets.splice(index, 1);
}
}
// Check for collision with enemies
for (var i = 0; i < enemies.length; i++) {
if (self.intersects(enemies[i]) && enemies[i].isFullyOnScreen()) {
enemies[i].takeDamage();
if (self.hitbox) {
if (self.hitbox.hitboxGraphics) {
self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when bullet is destroyed upon collision
}
self.hitbox.destroy(); // Ensure hitbox is removed when bullet is destroyed upon collision
}
self.destroy();
break;
}
}
};
});
// CityTiles class
var CityTiles = Container.expand(function () {
var self = Container.call(this);
// Attach the city tiles sprite
var cityTilesSprite = self.attachAsset('CityTiles', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 4,
scaleY: 4
});
// Set city tiles speed
self.speed = 0.1;
// This is automatically called every game tick, if the city tiles are attached!
self.update = function () {
self.y += self.speed;
// Reset the position if the city tiles go off the screen
if (self.y > 2732 + self.height / 2) {
self.y = -self.height / 2;
}
};
});
// Cloud class
var Cloud = Container.expand(function () {
var self = Container.call(this);
// Randomly assign cloud asset
var cloudAssets = ['cloud1', 'cloud2', 'cloud3', 'cloud4', 'cloud5'];
var randomCloud = cloudAssets.splice(Math.floor(Math.random() * cloudAssets.length), 1)[0];
var scale = Math.random() * 0.5 + 0.5;
// Mirror cloud sprites 50% of the time
var flipX = Math.random() > 0.5 ? -1 : 1;
var cloudSprite = self.attachAsset(randomCloud, {
anchorX: 0.5,
anchorY: 0.5,
scaleX: scale * flipX,
scaleY: scale
});
// Set cloud sprite to be 70% transparent
cloudSprite.alpha = 0.1;
// Set cloud speed
self.speed = Math.random() * 3 + 2;
// This is automatically called every game tick, if the cloud is attached!
self.update = function () {
self.y += self.speed * 2 * (1 + self.x / 2048 * 0.5);
// Remove the cloud if it goes off the screen
if (self.y > 2732 + self.height / 2) {
// Remove the cloud from the clouds array
var index = clouds.indexOf(self);
if (index > -1) {
clouds.splice(index, 1);
}
cloudAssets.push(randomCloud); // Add the cloud back to the cloudAssets array
// Destroy the cloud
self.destroy();
}
};
});
// Enemy class
var Enemy = Container.expand(function () {
var self = Container.call(this);
self.isFullyOnScreen = function () {
return self.x - self.width / 2 >= 0 && self.x + self.width / 2 <= 2048 && self.y - self.height / 2 >= 0 && self.y + self.height / 2 <= 2732;
};
// This is automatically called every game tick, if the enemy is attached!
var enemyColors = [0xCC6666, 0x66CC66, 0x6666CC, 0xCCCC66, 0xCC66CC, 0x66CCCC, 0xCC9966, 0x99CC66, 0x6699CC, 0x9966CC]; // Define a more varied palette of colors
var randomColor = enemyColors[Math.floor(Math.random() * enemyColors.length)]; // Randomly select a color
var enemySprite = self.attachAsset('enemy', {
anchorX: 0.5,
anchorY: 0.5,
tint: randomColor // Apply the random color as a tint
});
// Create a hitbox for collision detection
self.hitbox = new Container();
self.hitbox.width = enemySprite.width * 0.5;
self.hitbox.height = enemySprite.height * 0.5;
self.hitbox.x = -self.hitbox.width / 2; // Center the hitbox
self.hitbox.y = -self.hitbox.height / 2; // Center the hitbox
self.addChild(self.hitbox);
// Draw the hitbox for visualization
var hitboxGraphics = LK.getAsset('enemy', {
width: self.hitbox.width,
height: self.hitbox.height,
color: 0xff0000,
// Red color for hitbox
shape: 'box'
});
self.hitbox.hitboxGraphics = hitboxGraphics;
self.hitbox.addChild(self.hitbox.hitboxGraphics);
// Set enemy speed
self.speed = 2;
// Initialize hit points
self.hitPoints = 3;
// Method to handle enemy damage
self.takeDamage = function () {
self.hitPoints -= 1;
// Flash enemy yellow when hit
tween(self, {
tint: 0xFFFF00
}, {
duration: 100,
easing: tween.linear,
onFinish: function onFinish() {
tween(self, {
tint: randomColor // Return to original random color
}, {
duration: 100,
easing: tween.linear
});
}
});
if (self.hitPoints <= 0) {
if (self.hitbox) {
self.hitbox.destroy(); // Ensure hitbox is removed when enemy is destroyed
}
self.destroy();
score += 10; // Increase score by 10 for each enemy destroyed
if (typeof scoreTxt !== 'undefined') {
scoreTxt.setText(score.toString()); // Update score display
if (score > highScore) {
highScore = score;
storage.highScore = highScore; // Persist new high score
}
}
// Ensure the enemy container is removed from the enemies array
var index = enemies.indexOf(self);
if (index > -1) {
enemies.splice(index, 1);
}
}
};
// This is automatically called every game tick, if the enemy is attached!
self.update = function () {
self.y += self.speed;
if (typeof self.speedX !== 'undefined') {
self.x += self.speedX; // Move horizontally if speedX is defined
if (self.x < 0 || self.x > 2048) {
self.speedX *= -1; // Reverse direction if hitting screen edge
}
}
// Check for overlap with other enemies and adjust position
for (var i = 0; i < enemies.length; i++) {
if (enemies[i] !== self) {
var dx = self.x - enemies[i].x;
var dy = self.y - enemies[i].y;
var distance = Math.sqrt(dx * dx + dy * dy);
if (distance < self.width) {
if (dx < 0) {
self.x -= 1; // Move left to avoid overlap
} else {
self.x += 1; // Move right to avoid overlap
}
}
}
}
// Implement dodging behavior
if (typeof playerVehicle !== 'undefined') {
if (self.x < playerVehicle.x && self.x < 2048 - self.width / 2) {
self.x += 1; // Move right to dodge
} else if (self.x > playerVehicle.x && self.x > self.width / 2) {
self.x -= 1; // Move left to dodge
}
}
// Destroy the enemy if it goes off the screen
if (self.y > 2732 + self.height) {
self.destroy();
}
// Fire a projectile every 180 ticks
if (typeof self.lastTick === 'undefined') {
self.lastTick = LK.ticks;
}
if (LK.ticks % 180 === 0) {
var projectile = new Projectile();
projectile.x = self.x;
projectile.y = self.y + self.height / 2 + projectile.height / 2;
game.addChild(projectile);
self.lastTick = LK.ticks;
}
};
});
// PlayerLifeBar class
var PlayerLifeBar = Container.expand(function () {
var self = Container.call(this);
// Create a background bar
var backgroundBar = self.attachAsset('playerHealthBackground', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 1.0,
scaleY: 1.0
});
backgroundBar.alpha = 0.5; // Set semi-transparency
// Create an array to hold the life segments
self.lifeSegments = [];
for (var i = 0; i < 10; i++) {
var lifeSegment = self.attachAsset('playerHealthBar', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 1.0,
// Keep original scale for life segments
scaleY: 1.0,
color: 0x00ff00,
// Green for life
alpha: 0.5 // Set semi-transparency
});
lifeSegment.x = (i - 4.5) * 100; // Position segments from left to right with 100px width
self.lifeSegments.push(lifeSegment);
}
// Method to update the life bar based on player's hit points
self.updateLife = function (hitPoints) {
for (var i = 0; i < self.lifeSegments.length; i++) {
self.lifeSegments[i].visible = i < hitPoints; // Show segments based on hit points
}
};
});
// Player's vehicle class
var PlayerVehicle = Container.expand(function () {
var self = Container.call(this);
// Attach the player's vehicle sprite
var vehicleSprite = self.attachAsset('playerVehicle', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 2,
scaleY: 2
});
// This is automatically called every game tick, if the player's vehicle is attached!
self.update = function () {
// Fire a bullet every 30 ticks
if (LK.ticks % 30 == 0) {
var bullet = new Bullet();
bullet.x = self.x;
bullet.y = self.y - self.height / 2 - bullet.height / 2;
game.addChild(bullet);
bullets.push(bullet);
}
};
// Add touch event handlers to the player's vehicle to enable dragging
self.down = function (x, y, obj) {
// Store the initial touch position
self.initialTouchPosition = {
x: x,
y: y
};
// Store the initial position of the player's vehicle
self.initialPosition = {
x: self.x,
y: self.y
};
};
self.move = function (x, y, obj) {
// Update the position of the player's vehicle relative to the speed and direction at which the player moves their finger
// Calculate the intended movement
var intendedX = self.initialPosition.x + (x - self.initialTouchPosition.x);
var intendedY = self.initialPosition.y + (y - self.initialTouchPosition.y);
// Calculate the distance to move
var deltaX = intendedX - self.x;
var deltaY = intendedY - self.y;
// Calculate the speed limit
var maxSpeed = 10; // Maximum speed per frame
// Calculate the actual movement based on the speed limit
var distance = Math.sqrt(deltaX * deltaX + deltaY * deltaY);
if (distance > maxSpeed) {
var ratio = maxSpeed / distance;
deltaX *= ratio;
deltaY *= ratio;
}
// Update the position with the limited speed
self.x += deltaX;
self.y += deltaY;
};
self.up = function (x, y, obj) {
// Reset the initial touch position and the initial position of the player's vehicle
self.initialTouchPosition = null;
self.initialPosition = null;
};
// Method to handle player damage
self.hitPoints = 10; // Initialize player hit points
self.takeDamage = function () {
self.hitPoints -= 1;
if (typeof playerLifeBar !== 'undefined') {
playerLifeBar.updateLife(self.hitPoints); // Update the life bar
if (self.hitPoints < playerLifeBar.lifeSegments.length) {
playerLifeBar.lifeSegments[self.hitPoints].visible = false; // Hide the rightmost segment
}
}
// Blink effect using tween
tween(self, {
alpha: 0.5
}, {
duration: 100,
easing: tween.linear,
onFinish: function onFinish() {
tween(self, {
alpha: 1
}, {
duration: 100,
easing: tween.linear
});
}
});
if (self.hitPoints <= 0) {
self.destroy(); // Destroy the player vehicle if hit points reach zero
LK.showGameOver(); // Trigger game over when player health is depleted
}
};
});
// Projectile class
var Projectile = Container.expand(function () {
var self = Container.call(this);
// Attach the projectile sprite
var projectileSprite = self.attachAsset('projectile', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 1.0,
// Keep original scale for sprite
scaleY: 1.0,
// Keep original scale for sprite
width: 100,
// Set sprite width to 100
height: 100 // Set sprite height to 100
});
// Create a hitbox for collision detection
var hitbox = new Container();
hitbox.width = projectileSprite.width * 0.5;
hitbox.height = projectileSprite.height * 0.5;
hitbox.x = -hitbox.width / 2; // Center the hitbox
hitbox.y = -hitbox.height / 2; // Center the hitbox
self.addChild(hitbox);
// Draw the hitbox for visualization
var hitboxGraphics = LK.getAsset('projectile', {
width: hitbox.width,
height: hitbox.height,
color: 0xff0000,
// Red color for hitbox
shape: 'box'
});
hitbox.addChild(hitboxGraphics);
// Set projectile speed
self.speed = 5;
// This is automatically called every game tick, if the projectile is attached!
self.update = function () {
if (self.lastX === undefined) {
self.lastX = self.x;
}
if (self.lastY === undefined) {
self.lastY = self.y;
}
self.y += self.speed;
self.lastX = self.x;
self.lastY = self.y;
// Destroy the projectile if it goes off the screen
if (self.y > 2732 + self.height / 2) {
self.destroy();
}
// Check for collision with player
// Ensure playerVehicle is defined before checking intersection
if (typeof playerVehicle !== 'undefined' && self.intersects(playerVehicle.hitbox)) {
// Damage the player
playerVehicle.takeDamage();
// Destroy the projectile
self.destroy();
if (self.hitbox) {
self.hitbox.destroy(); // Ensure hitbox is removed when projectile is destroyed
if (self.hitbox.hitboxGraphics) {
self.hitbox.hitboxGraphics.destroy(); // Ensure hitbox graphics are removed when projectile is destroyed
}
}
}
// Check for collision with bullets
for (var i = 0; i < bullets.length; i++) {
var dx = self.x - bullets[i].x;
var dy = self.y - bullets[i].y;
var distance = Math.sqrt(dx * dx + dy * dy);
if (distance < self.width / 2 + bullets[i].width / 2) {
// Allow bullets to pass through projectiles without being destroyed
continue;
}
}
};
});
/****
* Initialize Game
****/
// Add player's vehicle to the game
var game = new LK.Game({
backgroundColor: 0x000000
});
/****
* Game Code
****/
// Initialize score and display it at the top of the screen
var score = 0;
var highScore = storage.highScore || 0;
var scoreTxt = new Text2(score.toString(), {
size: 100,
fill: 0xFFFFFF
});
scoreTxt.anchor.set(0.5, 0);
var highScoreTxt = new Text2("High Score: " + highScore.toString(), {
size: 50,
fill: 0xFFFFFF
});
highScoreTxt.anchor.set(0.5, 0);
highScoreTxt.y = scoreTxt.height; // Position below the score
LK.gui.top.addChild(highScoreTxt);
LK.gui.top.addChild(scoreTxt);
// Create an array to store the bullets
var bullets = [];
// Create an array to store the enemies
var enemies = [];
// Create a timer to add enemies in waves
var enemyWaveTimer = LK.setInterval(function () {
// Add a wave of 3 enemies to reduce on-screen enemies
for (var i = 0; i < 2; i++) {
var enemy = game.addChild(new Enemy());
enemy.y = -enemy.height;
enemy.x = (i + 1) * (2048 / 6);
// Add the new enemy to the enemies array
enemies.push(enemy);
}
// Add diagonal spawn pattern
for (var i = 0; i < 2; i++) {
var enemy = game.addChild(new Enemy());
enemy.y = -enemy.height - i * 100; // Staggered start positions
enemy.x = i % 2 === 0 ? 0 : 2048; // Alternate starting from left and right
enemy.speedX = i % 2 === 0 ? 1 : -1; // Move right if starting from left, left if starting from right
enemies.push(enemy);
}
// Set a new random interval for the next wave
LK.clearInterval(enemyWaveTimer);
enemyWaveTimer = LK.setInterval(arguments.callee, Math.random() * 2000 + 4000); // Random interval between 4 and 6 seconds
}, Math.random() * 2000 + 4000); // Initial random interval between 4 and 6 seconds
// Increase enemy speed and spawn rate every time score reaches a multiple of 200
if (score % 200 === 0 && score !== 0) {
enemies.forEach(function (enemy) {
enemy.speed += 0.5; // Increase enemy speed
});
LK.clearInterval(enemyWaveTimer);
enemyWaveTimer = LK.setInterval(function () {
for (var i = 0; i < 3; i++) {
var enemy = game.addChild(new Enemy());
enemy.y = -enemy.height;
enemy.x = (i + 1) * (2048 / 6);
enemies.push(enemy);
}
// Add zigzag spawn pattern
for (var i = 0; i < 2; i++) {
var enemy = game.addChild(new Enemy());
enemy.y = -enemy.height;
enemy.x = i % 2 === 0 ? 0 : 2048; // Start from left or right
enemy.speedX = i % 2 === 0 ? 2 : -2; // Move in zigzag pattern
enemies.push(enemy);
}
}, Math.max(1000, 5000 - score / 200 * 500)); // Decrease interval time
}
// Add CityTiles to the game and repeat to fill the screen horizontally
var cityTilesArray = [];
for (var i = 0; i < Math.ceil(2048 / 804) + 1; i++) {
// 804 is the new width of a single CityTile
for (var j = 0; j < Math.ceil(2732 / 804) + 1; j++) {
// 804 is the new height of a single CityTile
var cityTile = game.addChild(new CityTiles());
cityTile.x = i * 804 - 402; // Adjust position for seamless tiling
cityTile.y = j * 804 - 402; // Adjust position for seamless tiling
cityTilesArray.push(cityTile);
}
}
// Add player's vehicle to the game
var playerVehicle = game.addChild(new PlayerVehicle());
playerVehicle.hitbox = new Container();
playerVehicle.hitbox.width = playerVehicle.width * 0.5; // Define hitbox size
playerVehicle.hitbox.height = playerVehicle.height * 0.5; // Define hitbox size
playerVehicle.hitbox.x = -playerVehicle.hitbox.width / 2; // Center the hitbox
playerVehicle.hitbox.y = -playerVehicle.hitbox.height / 2; // Center the hitbox
playerVehicle.addChild(playerVehicle.hitbox);
// Draw the hitbox for visualization
var playerHitboxGraphics = LK.getAsset('playerHitbox', {
width: playerVehicle.hitbox.width,
height: playerVehicle.hitbox.height,
color: 0x00ff00,
// Green color for hitbox
shape: 'box'
});
playerVehicle.hitbox.addChild(playerHitboxGraphics);
playerVehicle.x = 2048 / 2;
playerVehicle.y = 2732 - playerVehicle.height / 2;
// Add player's life bar to the game
var playerLifeBar = game.addChild(new PlayerLifeBar());
playerLifeBar.x = playerLifeBar.width / 2 + 10; // Position the life bar at the lower left corner with a 10 pixel margin
playerLifeBar.y = 2732 - playerLifeBar.height / 2 - 10; // Adjust y position to be at the bottom with a 10 pixel margin
var dragNode = null;
game.down = function (x, y, obj) {
dragNode = playerVehicle;
dragNode.initialTouchPosition = {
x: x,
y: y
};
dragNode.initialPosition = {
x: playerVehicle.x,
y: playerVehicle.y
};
};
game.move = function (x, y, obj) {
if (dragNode && typeof playerVehicle !== 'undefined') {
var newX = dragNode.initialPosition.x + (x - dragNode.initialTouchPosition.x);
var newY = dragNode.initialPosition.y + (y - dragNode.initialTouchPosition.y);
// Limit player vehicle movement to within the screen
newX = Math.max(Math.min(newX, 2048 - dragNode.width / 2), dragNode.width / 2);
newY = Math.max(Math.min(newY, 2732 - dragNode.height / 2), dragNode.height / 2);
dragNode.x = newX;
dragNode.y = newY;
}
};
game.up = function (x, y, obj) {
if (dragNode) {
dragNode = null;
}
};
// Create an array to store the clouds
var clouds = [];
// Create a timer to add clouds at random intervals between 1 and 5 seconds
var cloudTimer = LK.setInterval(function () {
// Check if there are less than 4 clouds on screen
if (clouds.length < 4) {
var cloud = game.addChild(new Cloud());
cloud.y = -cloud.height;
cloud.x = Math.random() * 2048;
// Add the new cloud to the clouds array
clouds.push(cloud);
}
}, Math.random() * 4000 + 1000);
Neon cyberpunk flying car. Single Game Texture. Blank background. High contrast. No shadows. Top-down
Fluffy cloud viewed from the top. Single Game Sprite Texture. No background.
This type of cloud but in a fine mist
One single Fluffy cloud viewed from the top. Single Game Sprite. No background. No shadow
Mini rocket projectile from a video game. No background.
Terminator style 2D enemy ship. Facing dead south. Single Game Texture. In-Game asset. 2d. Blank background. High contrast. No shadows
Circular ball projectile from a video game. No background.
Make it look like night time