User prompt
again
User prompt
move it again by 10 units
User prompt
again
User prompt
again
User prompt
Move the buy rainbow bubble button left by 20 units
User prompt
Set the ads opacity to 100%
User prompt
Shrink the ad by 10%
User prompt
again
User prompt
shrink the ad by 10%
User prompt
Make it so the as fills the screen
User prompt
Shrink the ad
User prompt
Disable user input during the ad
User prompt
Centre the ad.
User prompt
Centre the ad
User prompt
Layer the ad ontop of everything
User prompt
Make the ‘ad’ image big enough so it covers the entire screen
User prompt
Remove the other image in the ad
User prompt
Make the background of the ad use the ad image instead
User prompt
Make the ad use the ‘ad’ image
User prompt
make the text white
User prompt
move it down by 50 units
User prompt
again
User prompt
Again but by 100 units
/****
* Plugins
****/
var tween = LK.import("@upit/tween.v1");
/****
* Classes
****/
// Bubble class
var Bubble = Container.expand(function () {
var self = Container.call(this);
// Properties
self.color = null; // 'red', 'green', etc.
self.gridRow = null;
self.gridCol = null;
self.isMoving = false; // True if this bubble is flying (shot)
self.radius = 60; // Half of asset width/height
// Attach asset
self.setColor = function (color) {
self.color = color;
if (self.bubbleAsset) {
self.removeChild(self.bubbleAsset);
}
var assetId = color === 'rainbow' ? 'bubble_rainbow' : 'bubble_' + color;
self.bubbleAsset = self.attachAsset(assetId, {
anchorX: 0.5,
anchorY: 0.5
});
};
// For flying bubbles, set velocity
self.vx = 0;
self.vy = 0;
// For popping animation
self.pop = function (_onFinish) {
tween(self, {
scaleX: 0,
scaleY: 0,
alpha: 0
}, {
duration: 200,
easing: tween.easeIn,
onFinish: function onFinish() {
if (_onFinish) _onFinish();
}
});
};
// Update for moving bubbles
self.update = function () {
if (self.isMoving) {
self.x += self.vx;
self.y += self.vy;
}
};
return self;
});
// Cannon class
var Cannon = Container.expand(function () {
var self = Container.call(this);
// Attach cannon asset
self.cannonAsset = self.attachAsset('cannon', {
anchorX: 0.5,
anchorY: 0.5
});
// Angle in radians (0 = up)
self.angle = 0;
// Set angle and rotate cannon
self.setAngle = function (angle) {
self.angle = angle;
self.cannonAsset.rotation = angle;
};
return self;
});
/****
* Initialize Game
****/
var game = new LK.Game({
backgroundColor: 0x222244
});
/****
* Game Code
****/
// --- Game constants ---
// Bubble colors
// Cannon
// Bubble shoot sound
// Rainbow bubble: white color, acts as all colors
// New wall image asset
var GAME_W = 2048;
var GAME_H = 2732;
var BUBBLE_RADIUS = 60; // px
var BUBBLE_DIAM = 120;
var GRID_COLS = 12; // Number of columns
var GRID_ROWS = 14; // Number of rows
var GRID_TOP = 200; // px from top
var GRID_LEFT = (GAME_W - GRID_COLS * BUBBLE_DIAM) / 2; // Center grid
var COLORS = ['red', 'green', 'blue', 'yellow', 'purple'];
var SHOOTABLE_COLORS = ['red', 'green', 'blue', 'yellow', 'purple', 'rainbow'];
var SHOOT_SPEED = 38; // px per frame
var MIN_ANGLE = -Math.PI / 2 + Math.PI / 8; // -67.5 deg
var MAX_ANGLE = -Math.PI / 2 - Math.PI / 8; // -112.5 deg
// --- Game state ---
var grid = []; // 2D array [row][col] of Bubble or null
var flyingBubble = null; // The bubble currently being shot
var nextBubbleColor = null; // Color of next bubble
var cannon = null;
var score = 0;
var scoreTxt = null;
var isShooting = false;
var gameOver = false;
var isAdShowing = false; // Tracks if an ad overlay is active
// --- GUI ---
scoreTxt = new Text2('0', {
size: 120,
fill: 0xFFFFFF
});
scoreTxt.anchor.set(0.5, 0);
LK.gui.top.addChild(scoreTxt);
// --- Helper functions ---
// Get pixel position for grid cell
function gridToXY(row, col) {
var x = GRID_LEFT + col * BUBBLE_DIAM + BUBBLE_RADIUS;
var y = GRID_TOP + row * BUBBLE_DIAM + BUBBLE_RADIUS;
// Odd rows are offset (hex grid)
if (row % 2 === 1) x += BUBBLE_RADIUS;
return {
x: x,
y: y
};
}
// Get grid cell for pixel position
function xyToGrid(x, y) {
// Estimate row
var row = Math.round((y - GRID_TOP - BUBBLE_RADIUS) / BUBBLE_DIAM);
if (row < 0) row = 0;
if (row >= GRID_ROWS) row = GRID_ROWS - 1;
// Estimate col
var col = Math.round((x - GRID_LEFT - BUBBLE_RADIUS - (row % 2 === 1 ? BUBBLE_RADIUS : 0)) / BUBBLE_DIAM);
if (col < 0) col = 0;
if (col >= GRID_COLS) col = GRID_COLS - 1;
return {
row: row,
col: col
};
}
// Check if a grid cell is valid
function isValidCell(row, col) {
if (row < 0 || row >= GRID_ROWS) return false;
if (col < 0 || col >= GRID_COLS) return false;
return true;
}
// Get neighbors (hex grid)
function getNeighbors(row, col) {
// Even/odd row offset
var even = row % 2 === 0;
var neighbors = [{
row: row - 1,
col: col
},
// up
{
row: row - 1,
col: col + (even ? -1 : 1)
},
// up-left/up-right
{
row: row,
col: col - 1
},
// left
{
row: row,
col: col + 1
},
// right
{
row: row + 1,
col: col
},
// down
{
row: row + 1,
col: col + (even ? -1 : 1)
} // down-left/down-right
];
// Filter valid
var valid = [];
for (var i = 0; i < neighbors.length; ++i) {
var n = neighbors[i];
if (isValidCell(n.row, n.col)) valid.push(n);
}
return valid;
}
// Find all connected bubbles of the same color (DFS)
function findConnected(row, col, color, visited) {
if (!isValidCell(row, col)) return;
if (visited[row * GRID_COLS + col]) return;
var b = grid[row][col];
// Rainbow bubble matches any color, and any color matches rainbow
if (!b) return;
if (color === 'rainbow' || b.color === 'rainbow') {
// Allow connection
} else if (b.color !== color) {
return;
}
visited[row * GRID_COLS + col] = true;
var neighbors = getNeighbors(row, col);
for (var i = 0; i < neighbors.length; ++i) {
var n = neighbors[i];
findConnected(n.row, n.col, color, visited);
}
}
// Find all bubbles connected to the top (DFS)
function findConnectedToTop(visited) {
for (var col = 0; col < GRID_COLS; ++col) {
dfsTop(0, col, visited);
}
}
function dfsTop(row, col, visited) {
if (!isValidCell(row, col)) return;
if (visited[row * GRID_COLS + col]) return;
var b = grid[row][col];
if (!b) return;
visited[row * GRID_COLS + col] = true;
var neighbors = getNeighbors(row, col);
for (var i = 0; i < neighbors.length; ++i) {
var n = neighbors[i];
dfsTop(n.row, n.col, visited);
}
}
// Remove bubbles in visited
function removeBubbles(visited, onPop) {
var popped = 0;
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
if (visited[row * GRID_COLS + col]) {
var b = grid[row][col];
if (b) {
(function (bubble, row, col) {
bubble.pop(function () {
if (bubble.parent) bubble.parent.removeChild(bubble);
});
})(b, row, col);
grid[row][col] = null;
popped++;
}
}
}
}
if (popped > 0 && onPop) {
// Reset sound volume to 1.0 when used in non-bonus context
if (popped < 5) {
LK.getSound('bubble_pop').volume = 1.0;
}
LK.getSound('bubble_pop').play();
}
return popped;
}
// Check for game over (bubbles at bottom row)
function checkGameOver() {
for (var col = 0; col < GRID_COLS; ++col) {
if (grid[GRID_ROWS - 1][col]) {
return true;
}
}
return false;
}
// Check for win (all bubbles cleared)
function checkWin() {
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
if (grid[row][col]) return false;
}
}
return true;
}
// Get a random color from colors present on the board
function getRandomColorOnBoard() {
var present = {};
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
var b = grid[row][col];
if (b) present[b.color] = true;
}
}
var arr = [];
for (var i = 0; i < COLORS.length; ++i) {
if (present[COLORS[i]]) arr.push(COLORS[i]);
}
if (arr.length === 0) arr = COLORS.slice();
return arr[Math.floor(Math.random() * arr.length)];
}
// --- Game setup ---
// Initialize grid
function initGrid() {
grid = [];
for (var row = 0; row < GRID_ROWS; ++row) {
var arr = [];
for (var col = 0; col < GRID_COLS; ++col) {
arr.push(null);
}
grid.push(arr);
}
// Fill first 9 rows with random bubbles
for (var row = 0; row < 9; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
// Odd rows have one less bubble at the end
if (row % 2 === 1 && col === GRID_COLS - 1) continue;
var color = COLORS[Math.floor(Math.random() * COLORS.length)];
var bubble = new Bubble();
bubble.setColor(color);
var pos = gridToXY(row, col);
bubble.x = pos.x;
bubble.y = pos.y;
bubble.gridRow = row;
bubble.gridCol = col;
grid[row][col] = bubble;
game.addChild(bubble);
}
}
}
// Initialize cannon
function initCannon() {
cannon = new Cannon();
cannon.x = GAME_W / 2;
cannon.y = GAME_H - 180;
cannon.setAngle(-Math.PI / 2); // Up
game.addChild(cannon);
}
// Prepare next bubble color
function prepareNextBubble() {
// Only allow rainbow as a shootable bubble, and make it rare (1 in 30 chance)
if (Math.floor(Math.random() * 30) === 0) {
nextBubbleColor = 'rainbow';
} else {
// Pick from non-rainbow colors present on the board, or all if none
var present = {};
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
var b = grid[row][col];
if (b && b.color !== 'rainbow') present[b.color] = true;
}
}
var arr = [];
for (var i = 0; i < COLORS.length; ++i) {
if (present[COLORS[i]]) arr.push(COLORS[i]);
}
if (arr.length === 0) arr = COLORS.slice();
nextBubbleColor = arr[Math.floor(Math.random() * arr.length)];
}
// Always update preview bubble if it exists
if (previewBubble) {
previewBubble.setColor(nextBubbleColor);
}
}
// Launch a new flying bubble
function launchBubble(angle) {
if (isShooting || gameOver) return;
isShooting = true;
var bubble = new Bubble();
bubble.setColor(nextBubbleColor);
bubble.x = cannon.x;
bubble.y = cannon.y - 80;
bubble.isMoving = true;
// Set velocity
bubble.vx = Math.cos(angle) * SHOOT_SPEED;
bubble.vy = Math.sin(angle) * SHOOT_SPEED;
flyingBubble = bubble;
game.addChild(bubble);
LK.getSound('bubble_shoot').play();
prepareNextBubble();
}
// Snap flying bubble to grid
function snapBubbleToGrid(bubble) {
// Check if hitting left or right wall
var isWallHit = false;
var bestRow = 0,
bestCol = 0;
if (bubble.x <= BUBBLE_RADIUS + 5) {
// Left wall hit - find best row
isWallHit = true;
var bestDist = 99999;
for (var row = 0; row < GRID_ROWS; ++row) {
var pos = gridToXY(row, 0);
var dy = bubble.y - pos.y;
var dist = dy * dy;
if (dist < bestDist) {
bestDist = dist;
bestRow = row;
bestCol = 0;
}
}
} else if (bubble.x >= GAME_W - BUBBLE_RADIUS - 5) {
// Right wall hit - find best row
isWallHit = true;
var bestDist = 99999;
for (var row = 0; row < GRID_ROWS; ++row) {
var pos = gridToXY(row, GRID_COLS - 1);
var dy = bubble.y - pos.y;
var dist = dy * dy;
if (dist < bestDist) {
bestDist = dist;
bestRow = row;
bestCol = GRID_COLS - 1;
}
}
}
// If not a wall hit, find nearest empty cell normally
if (!isWallHit) {
var minDist = 99999;
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
// Odd rows have one less bubble at the end
if (row % 2 === 1 && col === GRID_COLS - 1) continue;
if (grid[row][col]) continue;
var pos = gridToXY(row, col);
var dx = bubble.x - pos.x;
var dy = bubble.y - pos.y;
var dist = dx * dx + dy * dy;
if (dist < minDist) {
minDist = dist;
bestRow = row;
bestCol = col;
}
}
}
}
// Place bubble
var pos = gridToXY(bestRow, bestCol);
bubble.x = pos.x;
bubble.y = pos.y;
bubble.isMoving = false;
bubble.vx = 0;
bubble.vy = 0;
bubble.gridRow = bestRow;
bubble.gridCol = bestCol;
grid[bestRow][bestCol] = bubble;
flyingBubble = null;
isShooting = false;
// Check for matches
var visited = {};
if (bubble.color === 'rainbow') {
// Rainbow pops the colored bubble it lands on (if any neighbor) and all connected of same color
var neighbors = getNeighbors(bestRow, bestCol);
var found = false;
for (var i = 0; i < neighbors.length; ++i) {
var n = neighbors[i];
var nb = grid[n.row][n.col];
if (nb && nb.color !== 'rainbow') {
// Find all connected bubbles of the same color as the neighboring bubble
findConnected(n.row, n.col, nb.color, visited);
// Also make the rainbow bubble pop itself
visited[bestRow * GRID_COLS + bestCol] = true;
found = true;
break; // Only use one colored neighbor as the seed
}
}
} else {
findConnected(bestRow, bestCol, bubble.color, visited);
}
// Count connected
var count = 0;
for (var k in visited) if (visited[k]) count++;
if (count >= 3 || bubble.color === 'rainbow' && count > 0) {
// Remove connected or single colored neighbor for rainbow
var popped = removeBubbles(visited, true);
// Apply bonus multiplier for popping 5+ bubbles at once
var bonus = 1.0;
if (popped >= 5) {
// 5 bubbles = 1.1x, 6 bubbles = 1.2x, etc.
bonus = 1.0 + (popped - 4) * 0.1;
// Set pop sound volume according to bonus
var soundVolume = Math.min(bonus, 2.0); // Cap at 200% volume
LK.getSound('bubble_pop').volume = soundVolume;
// Show bonus text
var bonusTxt = new Text2('BONUS x' + bonus.toFixed(1), {
size: 80,
fill: 0xFFFF00
});
bonusTxt.anchor.set(0.5, 0.5);
bonusTxt.x = bubble.x;
bonusTxt.y = bubble.y;
game.addChild(bonusTxt);
// Animate and remove bonus text
tween(bonusTxt, {
y: bonusTxt.y - 100,
alpha: 0
}, {
duration: 1000,
onFinish: function onFinish() {
if (bonusTxt.parent) bonusTxt.parent.removeChild(bonusTxt);
}
});
}
score += Math.floor(popped * 10 * bonus);
scoreTxt.setText(score);
// Remove unattached bubbles
var attached = {};
findConnectedToTop(attached);
var floating = {};
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
if (grid[row][col] && !attached[row * GRID_COLS + col]) {
floating[row * GRID_COLS + col] = true;
}
}
}
var dropped = removeBubbles(floating, true);
if (dropped > 0) {
score += dropped * 20;
scoreTxt.setText(score);
}
}
// Check for win/lose
if (bubble.color !== 'rainbow' && checkWin()) {
// --- Spawn a new group of bubbles with different colors ---
// Pick a new set of 5 colors different from the previous COLORS
var prevColors = COLORS.slice();
var allColors = ['red', 'green', 'blue', 'yellow', 'purple'];
// Shuffle and pick a new set that is not the same as previous
var newColors = [];
var attempts = 0;
while (attempts < 10) {
// Shuffle allColors
var shuffled = allColors.slice();
for (var i = shuffled.length - 1; i > 0; i--) {
var j = Math.floor(Math.random() * (i + 1));
var temp = shuffled[i];
shuffled[i] = shuffled[j];
shuffled[j] = temp;
}
newColors = shuffled.slice(0, 5);
// Check if at least 2 colors are different from previous
var diffCount = 0;
for (var i = 0; i < 5; ++i) {
if (prevColors.indexOf(newColors[i]) === -1) diffCount++;
}
if (diffCount > 0) break;
attempts++;
}
COLORS = newColors;
// Clear the grid and respawn new bubbles
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
if (grid[row][col]) {
if (grid[row][col].parent) grid[row][col].parent.removeChild(grid[row][col]);
grid[row][col] = null;
}
}
}
// Fill first 9 rows with new random bubbles
for (var row = 0; row < 9; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
if (row % 2 === 1 && col === GRID_COLS - 1) continue;
var color = COLORS[Math.floor(Math.random() * COLORS.length)];
var bubble = new Bubble();
bubble.setColor(color);
var pos = gridToXY(row, col);
bubble.x = pos.x;
bubble.y = pos.y;
bubble.gridRow = row;
bubble.gridCol = col;
grid[row][col] = bubble;
game.addChild(bubble);
}
}
// Prepare next bubble with new color set
prepareNextBubble();
gameOver = false;
return;
}
if (checkGameOver()) {
LK.effects.flashScreen(0xff0000, 1000);
LK.showGameOver();
gameOver = true;
return;
}
}
// --- Game setup ---
initGrid();
initCannon();
prepareNextBubble();
// --- Add left and right wall boxes ---
var wallThickness = 280;
var wallHeight = GAME_H;
var leftWall = LK.getAsset('wall_image', {
anchorX: 0,
anchorY: 0,
width: wallThickness,
height: wallHeight,
x: 0,
y: 0
});
var rightWall = LK.getAsset('wall_image', {
anchorX: 0,
anchorY: 0,
width: wallThickness,
height: wallHeight,
x: GAME_W - wallThickness,
y: 0
});
game.addChild(leftWall);
game.addChild(rightWall);
// --- Draw next bubble preview ---
// Create a container for the preview bubble and background
var previewContainer = new Container();
previewContainer.x = GAME_W - 470; // Moved another 50 units further left
previewContainer.y = GAME_H - 200; // Moved 'centerCircle' (previewBackground in previewContainer) down by another 50 units
// Add a circular background behind the preview bubble
var previewBackground = LK.getAsset('centerCircle', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: (100 + 30) / 100 * 2.5,
// Increase by 30 units relative to original 100px asset
scaleY: (100 + 30) / 100 * 2.5,
alpha: 0.3,
tint: 0xFFFFFF
});
previewContainer.addChild(previewBackground);
// Create preview bubble inside the container
var previewBubble = new Bubble();
previewBubble.x = 0; // Center in container
previewBubble.y = 30; // Move preview bubble down by 10 more units
// Set the initial preview bubble color
if (nextBubbleColor) {
previewBubble.setColor(nextBubbleColor);
} else {
// If nextBubbleColor isn't set yet, use a default color
previewBubble.setColor(COLORS[Math.floor(Math.random() * COLORS.length)]);
}
previewContainer.addChild(previewBubble);
game.addChild(previewContainer);
// --- Add rainbow bubble button ---
var rainbowButton = new Container();
var rainbowBubble = LK.getAsset('bubble_rainbow', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 1.2,
scaleY: 1.2
});
rainbowButton.addChild(rainbowBubble);
// Add price text
var rainbowPriceTxt = new Text2('500', {
size: 60,
fill: 0xFFFFFF
});
rainbowPriceTxt.anchor.set(0.5, 0.5);
rainbowPriceTxt.y = 80;
rainbowButton.addChild(rainbowPriceTxt);
// Position button on the left side of the screen
rainbowButton.x = 120;
rainbowButton.y = GAME_H - 120;
game.addChild(rainbowButton);
// Add button press events
rainbowButton.interactive = true;
rainbowButton.down = function (x, y, obj) {
if (gameOver || isShooting) return;
// Check if player has enough points
if (score >= 500) {
// Subtract points
score -= 500;
scoreTxt.setText(score);
// Set next bubble to rainbow and make it immediately available
if (flyingBubble === null) {
// If no bubble is flying, make the next bubble rainbow
nextBubbleColor = 'rainbow';
previewBubble.setColor(nextBubbleColor);
} else {
// If a bubble is currently flying, queue up rainbow as the next one
nextBubbleColor = 'rainbow';
previewBubble.setColor(nextBubbleColor);
}
// Visual feedback
tween(rainbowButton, {
scaleX: 0.8,
scaleY: 0.8
}, {
duration: 100,
onFinish: function onFinish() {
tween(rainbowButton, {
scaleX: 1,
scaleY: 1
}, {
duration: 100
});
}
});
} else {
// Visual feedback for not enough points
tween(rainbowPriceTxt, {
scaleX: 1.3,
scaleY: 1.3,
tint: 0xFF0000
}, {
duration: 200,
onFinish: function onFinish() {
tween(rainbowPriceTxt, {
scaleX: 1,
scaleY: 1,
tint: 0xFFFFFF
}, {
duration: 200
});
}
});
}
};
// --- Add 'Watch Ad for 250 Score' button ---
var adButton = new Container();
// Use the unique 'ad_button' image asset as the button background
var adBg = LK.getAsset('ad_button', {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 1.5333333333333332,
scaleY: 1.5333333333333332,
alpha: 1
});
adButton.addChild(adBg);
var adTxt = new Text2('Watch Ad\n+250', {
size: 35,
fill: 0xFFFFFF,
align: 'center'
});
adTxt.anchor.set(0.5, 0.5);
adTxt.y = -170;
adButton.addChild(adTxt);
// Position ad button to the right of the rainbow button, but not too close to the right edge
adButton.x = 125;
adButton.y = GAME_H - 370;
game.addChild(adButton);
adButton.interactive = true;
adButton.down = function (x, y, obj) {
if (gameOver || isShooting) return;
// Show a 10 second "ad" overlay, then give 250 score
var overlay = new Container();
var bg = LK.getAsset('ad', {
anchorX: 0.5,
anchorY: 0.5,
width: GAME_W * 0.9,
height: GAME_H * 0.9,
alpha: 1,
x: 0,
y: 0
});
overlay.addChild(bg);
var adMsg = new Text2('Ad playing...\n(10 seconds)', {
size: 90,
fill: 0xFFFFFF,
align: 'center'
});
adMsg.anchor.set(0.5, 0.5);
adMsg.x = 0;
adMsg.y = -60;
overlay.addChild(adMsg);
var timerTxt = new Text2('10', {
size: 120,
fill: 0xFFD700
});
timerTxt.anchor.set(0.5, 0.5);
timerTxt.y = 100;
overlay.addChild(timerTxt);
// Add overlay to LK.gui.center. The overlay's coordinate system origin (top-left)
// will now be relative to the screen's center point.
LK.gui.center.addChild(overlay);
isAdShowing = true; // Disable input while ad is showing
// Position the overlay's top-left corner at (0,0) relative to LK.gui.center.
// This effectively places the overlay's origin at the screen center.
overlay.x = 0;
overlay.y = 0;
// Since the ad background 'bg' is centered within 'overlay' (anchor 0.5,0.5 at overlay's 0,0),
// and text elements are also centered horizontally within 'overlay', the entire ad display
// will now appear centered on the screen.
var secondsLeft = 10;
var adTimer = null;
function tickAdTimer() {
secondsLeft--;
timerTxt.setText(secondsLeft.toString());
if (secondsLeft <= 0) {
LK.clearInterval(adTimer);
// Remove overlay
if (overlay.parent) overlay.parent.removeChild(overlay);
isAdShowing = false; // Re-enable input after ad
// Give 250 score
score += 250;
scoreTxt.setText(score);
// Visual feedback
tween(adButton, {
scaleX: 1.2,
scaleY: 1.2
}, {
duration: 120,
onFinish: function onFinish() {
tween(adButton, {
scaleX: 1,
scaleY: 1
}, {
duration: 120
});
}
});
}
}
timerTxt.setText(secondsLeft.toString());
adTimer = LK.setInterval(tickAdTimer, 1000);
};
// --- Input handling ---
// Drag to aim
var aiming = false;
var aimAngle = -Math.PI / 2; // Up
// For drawing the aim line
var aimLine = null;
var aimLineStart = {
x: 0,
y: 0
};
var aimLineEnd = {
x: 0,
y: 0
};
// Helper to create or update the aim line
function updateAimLine(startX, startY, endX, endY) {
// Remove old aimLine if exists
if (aimLine && aimLine.parent) {
aimLine.parent.removeChild(aimLine);
}
// Create a new Container for the line
aimLine = new Container();
// Draw the line using a series of small bubbles (dots) between start and end
var dx = endX - startX;
var dy = endY - startY;
var dist = Math.sqrt(dx * dx + dy * dy);
var steps = Math.floor(dist / 32);
// Create a continuous line effect using the bubble color that matches nextBubbleColor
var lineColor = nextBubbleColor || 'blue';
var lineBaseAsset = lineColor === 'rainbow' ? 'bubble_rainbow' : 'bubble_' + lineColor;
for (var i = 0; i <= steps; ++i) {
var t = i / steps;
var px = startX + dx * t;
var py = startY + dy * t;
// Use a small, semi-transparent bubble of the current color as a dot
var dot = LK.getAsset(lineBaseAsset, {
anchorX: 0.5,
anchorY: 0.5,
scaleX: 0.18,
scaleY: 0.18,
alpha: 0.35 + (i === 0 ? 0.2 : 0) // Make first dot slightly more visible
});
dot.x = px;
dot.y = py;
// Add subtle animation to the dots
tween(dot, {
alpha: dot.alpha - 0.1,
scaleX: dot.scaleX - 0.03,
scaleY: dot.scaleY - 0.03
}, {
duration: 300 + i * 10,
easing: tween.easeOut
});
aimLine.addChild(dot);
}
// Add to game
game.addChild(aimLine);
}
function clamp(val, min, max) {
if (val < min) return min;
if (val > max) return max;
return val;
}
game.down = function (x, y, obj) {
if (gameOver || isAdShowing) return;
aiming = true;
handleAim(x, y);
};
game.move = function (x, y, obj) {
if (isAdShowing) return; // Prevent input during ad
if (aiming && !isShooting && !gameOver) {
handleAim(x, y);
}
};
game.up = function (x, y, obj) {
if (isAdShowing) return; // Prevent input during ad
if (aiming && !isShooting && !gameOver) {
launchBubble(aimAngle);
}
aiming = false;
// Remove aim line when shot
if (aimLine && aimLine.parent) {
aimLine.parent.removeChild(aimLine);
aimLine = null;
}
};
function handleAim(x, y) {
// Calculate angle from cannon to (x, y)
var dx = x - cannon.x;
var dy = y - cannon.y;
var angle = Math.atan2(dy, dx);
// Clamp angle to allowed range
angle = clamp(angle, MAX_ANGLE, MIN_ANGLE);
aimAngle = angle;
cannon.setAngle(angle);
// Draw aim line from cannon tip to the aiming point
var startX = cannon.x + Math.cos(angle) * 100;
var startY = cannon.y + Math.sin(angle) * 100;
var endX = cannon.x + Math.cos(angle) * 1200;
var endY = cannon.y + Math.sin(angle) * 1200;
updateAimLine(startX, startY, endX, endY);
}
// --- Game update loop ---
game.update = function () {
// Update flying bubble
if (flyingBubble && flyingBubble.isMoving) {
flyingBubble.update();
// Stick to walls instead of bouncing
if (flyingBubble.x < BUBBLE_RADIUS) {
// Stick to left wall
flyingBubble.x = BUBBLE_RADIUS;
snapBubbleToGrid(flyingBubble);
}
if (flyingBubble.x > GAME_W - BUBBLE_RADIUS) {
// Stick to right wall
flyingBubble.x = GAME_W - BUBBLE_RADIUS;
snapBubbleToGrid(flyingBubble);
}
// Check collision with top
if (flyingBubble.y < GRID_TOP + BUBBLE_RADIUS) {
snapBubbleToGrid(flyingBubble);
} else {
// Check collision with grid bubbles
var hit = false;
for (var row = 0; row < GRID_ROWS; ++row) {
for (var col = 0; col < GRID_COLS; ++col) {
var b = grid[row][col];
if (!b) continue;
var dx = flyingBubble.x - b.x;
var dy = flyingBubble.y - b.y;
var dist = Math.sqrt(dx * dx + dy * dy);
if (dist < BUBBLE_DIAM - 2) {
snapBubbleToGrid(flyingBubble);
hit = true;
break;
}
}
if (hit) break;
}
// If bubble falls below bottom, game over
if (flyingBubble && flyingBubble.y > GAME_H - 100) {
LK.effects.flashScreen(0xff0000, 1000);
LK.showGameOver();
gameOver = true;
return;
}
}
}
// Update preview bubble color
if (previewBubble && nextBubbleColor && previewBubble.color !== nextBubbleColor) {
previewBubble.setColor(nextBubbleColor);
}
}; ===================================================================
--- original.js
+++ change.js
@@ -648,9 +648,9 @@
rainbowPriceTxt.anchor.set(0.5, 0.5);
rainbowPriceTxt.y = 80;
rainbowButton.addChild(rainbowPriceTxt);
// Position button on the left side of the screen
-rainbowButton.x = 130;
+rainbowButton.x = 120;
rainbowButton.y = GAME_H - 120;
game.addChild(rainbowButton);
// Add button press events
rainbowButton.interactive = true;