const FogOfWar = { // Set of revealed hex coordinates (as strings) revealed: new Set(), // Configuration VISION_RANGE: Config.player.VISION_RANGE, init() { this.revealed.clear(); this.updateVisibility(player.position); }, // Convert hex to string key for Set storage hexToKey(hex) { return `${hex.q},${hex.r}`; }, // Check if a hex is currently visible isVisible(hex) { const playerPos = player.getCurrentPosition(); const distance = this.getHexDistance(hex, playerPos); return distance <= this.VISION_RANGE; }, // Check if a hex has been revealed isRevealed(hex) { return this.revealed.has(this.hexToKey(hex)); }, // Calculate distance between two hexes getHexDistance(a, b) { return Math.max( Math.abs(a.q - b.q), Math.abs(a.r - b.r), Math.abs((a.q + a.r) - (b.q + b.r)) ); }, // Update visibility based on player position updateVisibility(center) { // Get all hexes within vision range for (let q = -this.VISION_RANGE; q <= this.VISION_RANGE; q++) { for (let r = -this.VISION_RANGE; r <= this.VISION_RANGE; r++) { const hex = { q: center.q + q, r: center.r + r }; if (this.getHexDistance(center, hex) <= this.VISION_RANGE) { this.revealed.add(this.hexToKey(hex)); } } } }, // Draw fog of war effect draw(ctx) { HexGrid.getViewportHexes().forEach(hex => { const pixel = HexGrid.toPixel(hex); const screenX = Math.round(pixel.x - Camera.x); const screenY = Math.round(pixel.y - Camera.y); if (!this.isRevealed(hex)) { ctx.fillStyle = 'rgba(0, 0, 0, 1)'; this.drawHexShape(ctx, screenX, screenY); } else if (!this.isVisible(hex)) { ctx.fillStyle = 'rgba(0, 0, 0, 0.5)'; this.drawHexShape(ctx, screenX, screenY); } }); }, // Helper method to draw hex shape drawHexShape(ctx, x, y) { ctx.beginPath(); for (let i = 0; i < 6; i++) { const angle = 2 * Math.PI / 6 * i; const xPos = Math.round(x + HexGrid.SIZE * Math.cos(angle)); const yPos = Math.round(y + HexGrid.SIZE * Math.sin(angle)); if (i === 0) { ctx.moveTo(xPos, yPos); } else { ctx.lineTo(xPos, yPos); } } ctx.closePath(); ctx.fill(); } };