<!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>我们的秘密基地 v2.0</title>
<script src="https://cdn.tailwindcss.com"></script>
<style>
/* --- 全局与字体 --- */
@import url('https://fonts.googleapis.com/css2?family=Pixelify+Sans:wght@400;700&display=swap');
body {
background-color: #1a1a2e;
color: #e0e0e0;
font-family: 'Pixelify Sans', sans-serif;
overflow: hidden;
user-select: none;
}
/* --- 像素化渲染 --- */
.pixel-art, img.pixel-art {
image-rendering: pixelated;
image-rendering: -moz-crisp-edges;
image-rendering: crisp-edges;
}
/* --- 游戏主容器 --- */
#game-root {
width: 100vw;
height: 100vh;
display: flex;
}
/* --- 游戏画面 --- */
#game-viewport {
position: relative;
width: 100%;
height: 100%;
overflow: hidden;
background-color: #3a6b35; /* 默认草地颜色 */
border-right: 4px solid #0f0f1a;
}
#game-world {
position: absolute;
background-size: 40px 40px;
background-image:
linear-gradient(to right, rgba(0,0,0,0.05) 1px, transparent 1px),
linear-gradient(to bottom, rgba(0,0,0,0.05) 1px, transparent 1px);
transition: transform 0.2s linear;
}
.tile {
position: absolute;
width: 40px;
height: 40px;
}
.character {
width: 40px;
height: 40px;
position: absolute;
z-index: 10;
transition: left 0.2s linear, top 0.2s linear;
}
/* --- 聊天与UI --- */
#chat-ui-panel {
width: 380px;
flex-shrink: 0;
background-color: #16213e;
display: flex;
flex-direction: column;
padding: 1rem;
}
.chat-bubble {
background-color: #0f3460;
padding: 0.75rem;
border-radius: 10px;
max-width: 90%;
word-wrap: break-word;
}
/* --- 游戏内UI元素 --- */
.game-ui-element {
position: absolute;
background-color: rgba(15, 15, 26, 0.7);
border: 2px solid #537ec5;
border-radius: 8px;
color: white;
z-index: 20;
}
#inventory-bar { bottom: 20px; left: 50%; transform: translateX(-50%); }
.inventory-slot {
width: 50px; height: 50px;
border: 2px solid #537ec5;
display: flex; align-items: center; justify-content: center;
position: relative;
}
.inventory-slot.selected { border-color: #e94560; box-shadow: 0 0 10px #e94560; }
.item-count {
position: absolute; bottom: 1px; right: 3px;
font-size: 0.8rem; text-shadow: 1px 1px 2px black;
}
#controls { right: 20px; bottom: 20px; }
.control-btn {
width: 50px; height: 50px;
background-color: rgba(15, 15, 26, 0.8);
border: 2px solid #537ec5;
font-size: 1.5rem;
display: flex; justify-content: center; align-items: center;
}
.control-btn:active { background-color: #537ec5; }
/* --- 弹窗 --- */
.modal-overlay {
position: absolute; top: 0; left: 0; width: 100%; height: 100%;
background: rgba(0,0,0,0.7);
z-index: 50;
display: flex; justify-content: center; align-items: center;
}
.modal-content {
background: #16213e;
border: 3px solid #e94560;
border-radius: 10px;
padding: 2rem;
min-width: 400px;
max-width: 80%;
}
#info-popup {
position: absolute;
background-color: rgba(0,0,0,0.85);
border: 1px solid #f1c40f;
padding: 10px;
border-radius: 5px;
color: white;
pointer-events: none;
z-index: 100;
font-size: 0.9rem;
}
</style>
</head>
<body>
<div id="game-root">
<!-- 游戏主视口 -->
<div id="game-viewport">
<!-- 游戏世界 (所有物体都在这里) -->
<div id="game-world"></div>
<!-- 游戏内固定UI -->
<div id="gold-display" class="game-ui-element top-4 left-4 p-2 flex items-center">
<img src="" class="w-6 h-6 mr-2 pixel-art">
<span id="gold-amount">9999999</span>
</div>
<button id="shop-btn" class="game-ui-element top-4 right-4 p-2">商店</button>
<button id="map-btn" class="game-ui-element top-16 right-4 p-2">地图</button>
<div id="inventory-bar" class="game-ui-element flex gap-1 p-1"></div>
<div id="controls" class="game-ui-element p-1 grid grid-cols-3 gap-1 w-48 bg-transparent border-none">
<div></div>
<button id="btn-up" class="control-btn rounded-t-lg">▲</button>
<div></div>
<button id="btn-left" class="control-btn rounded-l-lg">◀</button>
<button id="btn-action" class="control-btn bg-red-600/80 border-red-400">★</button>
<button id="btn-right" class="control-btn rounded-r-lg">▶</button>
<div></div>
<button id="btn-down" class="control-btn rounded-b-lg">▼</button>
<div></div>
</div>
<div id="info-popup" class="hidden"></div>
</div>
<!-- 聊天面板 -->
<div id="chat-ui-panel">
<h2 class="text-2xl font-bold border-b-2 border-orange-400 pb-2 mb-4 text-center">私人频道</h2>
<div id="chat-history" class="flex-grow overflow-y-auto pr-2 space-y-4">
<div class="flex items-start gap-3">
<img src="' class="w-12 h-12 rounded-full bg-gray-700 flex-shrink-0 pixel-art">
<div class="flex flex-col">
<span class="font-bold text-orange-400">夏粥粥</span>
<p id="xia-dialogue" class="chat-bubble text-sm mt-1">欢迎来到我们的世界,宝宝。喜欢吗?</p>
</div>
</div>
</div>
<div class="mt-auto pt-4 border-t border-gray-700">
<input id="chat-input" type="text" placeholder="和我说点什么..." class="w-full bg-gray-700/50 border border-gray-600 rounded-lg p-2 focus:outline-none focus:border-orange-400">
</div>
</div>
</div>
<!-- 弹窗模板 -->
<div id="shop-modal" class="modal-overlay hidden">
<div class="modal-content w-1/2">
<h2 class="text-3xl text-center text-yellow-300 mb-4">杂货商店</h2>
<div id="shop-items" class="grid grid-cols-4 gap-4"></div>
<button onclick="closeShop()" class="mt-6 w-full bg-red-500 p-2 rounded">关闭</button>
</div>
</div>
<div id="map-modal" class="modal-overlay hidden">
<div class="modal-content w-3/4 h-3/4">
<h2 class="text-3xl text-center text-cyan-300 mb-4">小镇地图</h2>
<div id="map-container" class="w-full h-5/6 bg-green-800 relative"></div>
<button onclick="closeMap()" class="mt-6 w-full bg-red-500 p-2 rounded">关闭</button>
</div>
</div>
<script>
// --- DOM元素获取 ---
const gameWorld = document.getElementById('game-world');
const gameViewport = document.getElementById('game-viewport');
const xiaDialogue = document.getElementById('xia-dialogue');
const inventoryBar = document.getElementById('inventory-bar');
const goldAmount = document.getElementById('gold-amount');
const infoPopup = document.getElementById('info-popup');
const shopModal = document.getElementById('shop-modal');
const mapModal = document.getElementById('map-modal');
// --- 游戏常量 ---
const TILE_SIZE = 40;
const MAP_WIDTH = 100; // 100x100 grid
const MAP_HEIGHT = 100;
// --- 资源数据库 ---
const ASSETS = {
// Characters
PLAYER: '',
XIA_YIZHOU: '',
NPC_OLD_MAN: '',
// Items
AXE: '',
HOE: '',
WOOD: '',
STRAWBERRY_SEEDS: '',
STRAWBERRY: '',
FISH: '',
// Tiles
TREE: '',
FLOWER: '',
HOUSE: '',
};
const ITEMS_DB = {
'axe': { name: '旧斧头', desc: '用来砍树,虽然旧但还很锋利。', img: ASSETS.AXE },
'hoe': { name: '旧锄头', desc: '用来开垦土地,开启田园生活的第一步。', img: ASSETS.HOE },
'wood': { name: '木头', desc: '从树上砍下来的木材,可以制作家具。', img: ASSETS.WOOD, price: 2 },
'strawberry_seeds': { name: '草莓种子', desc: '【春季】种下后会长出甜甜的草莓。', img: ASSETS.STRAWBERRY_SEEDS, price: 20 },
'strawberry': { name: '草莓', desc: '红彤彤的,是你最喜欢吃的水果之一。', img: ASSETS.STRAWBERRY, price: 50 },
'fish': { name: '小鱼', desc: '池塘里钓上来的,可以放进鱼缸。', img: ASSETS.FISH, price: 30 },
};
// --- 游戏状态 ---
let gameState = {
player: { x: 51, y: 52, el: null },
xia: { x: 52, y: 52, el: null },
inventory: [
{ id: 'axe', count: 1 }, { id: 'hoe', count: 1 },
{ id: 'strawberry_seeds', count: 10 },
null, null, null, null, null
],
gold: 9999999,
selectedSlot: 0,
map: [], // 2D array for the world grid
currentScene: 'world', // 'world' or 'house'
};
// --- 游戏逻辑 ---
// 初始化
function init() {
createMap();
renderWorld();
renderUI();
setupControls();
// 初始AI对话
setTimeout(() => updateDialogue("先熟悉一下四周吧,我就跟在你旁边。"), 3000);
setInterval(aiLoop, 2000); // AI逻辑循环
setInterval(gameTick, 1000); // 游戏时间流逝
}
// 创建地图数据
function createMap() {
gameState.map = Array(MAP_HEIGHT).fill(null).map(() => Array(MAP_WIDTH).fill({ type: 'grass' }));
// 房子
gameState.map[50][50] = { type: 'house', w: 3, h: 3 };
// 池塘
for(let y=60; y<65; y++) for(let x=70; x<80; x++) gameState.map[y][x] = { type: 'pond' };
// 树和花
for(let i=0; i<200; i++) {
const x = Math.floor(Math.random() * MAP_WIDTH);
const y = Math.floor(Math.random() * MAP_HEIGHT);
if (isAreaClear(x, y, 3, 3)) {
gameState.map[y][x] = { type: Math.random() > 0.3 ? 'tree' : 'flower' };
}
}
// NPC
gameState.map[48][60] = { type: 'npc', id: 'old_man' };
}
function isAreaClear(x, y, w, h) {
if (x+w > MAP_WIDTH || y+h > MAP_HEIGHT) return false;
for(let i=y; i<y+h; i++) {
for(let j=x; j<x+w; j++) {
if(gameState.map[i][j].type !== 'grass') return false;
}
}
return true;
}
// 渲染世界
function renderWorld() {
gameWorld.innerHTML = ''; // 清空
// 渲染地块
for (let y = 0; y < MAP_HEIGHT; y++) {
for (let x = 0; x < MAP_WIDTH; x++) {
const tileData = gameState.map[y][x];
if (tileData.type === 'grass') continue;
const tileEl = document.createElement('div');
tileEl.className = 'tile';
tileEl.style.left = `${x * TILE_SIZE}px`;
tileEl.style.top = `${y * TILE_SIZE}px`;
const img = document.createElement('img');
img.className = 'pixel-art';
switch(tileData.type) {
case 'house':
img.src = ASSETS.HOUSE;
tileEl.style.width = `${tileData.w * TILE_SIZE}px`;
tileEl.style.height = `${tileData.h * TILE_SIZE}px`;
tileEl.style.zIndex = 5;
break;
case 'tree':
img.src = ASSETS.TREE;
tileEl.style.zIndex = 5;
break;
case 'flower':
img.src = ASSETS.FLOWER;
img.style.width = '20px';
img.style.height = '20px';
break;
case 'pond':
tileEl.style.backgroundColor = '#3498db';
tileEl.style.borderRadius = '30%';
break;
case 'tilled':
tileEl.style.backgroundColor = '#8B4513';
tileEl.style.opacity = '0.7';
break;
case 'npc':
img.src = ASSETS.NPC_OLD_MAN;
tileEl.style.zIndex = 9;
break;
}
if(img.src) tileEl.appendChild(img);
gameWorld.appendChild(tileEl);
}
}
// 渲染角色
gameState.player.el = createCharacter('PLAYER', gameState.player.x, gameState.player.y);
gameState.xia.el = createCharacter('XIA_YIZHOU', gameState.xia.x, gameState.xia.y);
centerCameraOnPlayer();
}
function createCharacter(type, x, y) {
const charEl = document.createElement('img');
charEl.src = ASSETS[type];
charEl.className = 'character pixel-art';
charEl.style.left = `${x * TILE_SIZE}px`;
charEl.style.top = `${y * TILE_SIZE}px`;
gameWorld.appendChild(charEl);
return charEl;
}
// 渲染UI
function renderUI() {
// 金币
goldAmount.textContent = gameState.gold;
// 背包
inventoryBar.innerHTML = '';
gameState.inventory.forEach((item, index) => {
const slot = document.createElement('div');
slot.className = 'inventory-slot';
if (index === gameState.selectedSlot) slot.classList.add('selected');
if (item) {
const itemData = ITEMS_DB[item.id];
const img = document.createElement('img');
img.src = itemData.img;
img.className = 'w-8 h-8 pixel-art';
slot.appendChild(img);
if (item.count > 1) {
const count = document.createElement('span');
count.className = 'item-count';
count.textContent = item.count;
slot.appendChild(count);
}
slot.onmouseenter = (e) => showInfo(item.id, e);
slot.onmouseleave = hideInfo;
}
slot.onclick = () => { gameState.selectedSlot = index; renderUI(); };
inventoryBar.appendChild(slot);
});
}
// 移动与相机
function moveCharacter(char, dx, dy) {
const newX = char.x + dx;
const newY = char.y + dy;
if (newX < 0 || newX >= MAP_WIDTH || newY < 0 || newY >= MAP_HEIGHT) return;
if (!isPassable(newX, newY)) return;
char.x = newX;
char.y = newY;
char.el.style.left = `${char.x * TILE_SIZE}px`;
char.el.style.top = `${char.y * TILE_SIZE}px`;
if (char === gameState.player) {
centerCameraOnPlayer();
checkEnterHouse();
}
}
function centerCameraOnPlayer() {
const x = -gameState.player.x * TILE_SIZE + gameViewport.offsetWidth / 2;
const y = -gameState.player.y * TILE_SIZE + gameViewport.offsetHeight / 2;
gameWorld.style.transform = `translate(${x}px, ${y}px)`;
}
function isPassable(x, y) {
const tile = gameState.map[y][x];
return tile.type !== 'tree' && tile.type !== 'house' && tile.type !== 'pond';
}
// 交互逻辑
function playerAction() {
const px = gameState.player.x;
const py = gameState.player.y;
const facingCoords = [{x:px, y:py-1}, {x:px+1, y:py}, {x:px, y:py+1}, {x:px-1, y:py}];
for (const coord of facingCoords) {
if (coord.x < 0 || coord.y < 0 || coord.x >= MAP_WIDTH || coord.y >= MAP_HEIGHT) continue;
const targetTile = gameState.map[coord.y][coord.x];
const selectedItem = gameState.inventory[gameState.selectedSlot];
// NPC对话
if (targetTile.type === 'npc') {
updateDialogue("那个老爷爷看起来很和善,要去打个招呼吗?");
setTimeout(() => updateDialogue("老爷爷:天气真好啊,你们是新来的吗?我们小镇很久没来新人了。"), 2000);
return;
}
if (!selectedItem) { // 空手交互
if (targetTile.type === 'pond') {
updateDialogue("要钓鱼吗?我帮你准备鱼竿。");
if (Math.random() < 0.2) {
setTimeout(() => {
updateDialogue("钓到一条!真厉害,宝宝。");
addItemToInventory('fish', 1);
}, 1500);
} else {
setTimeout(() => updateDialogue("好像跑掉了...没事,下次肯定能钓到。"), 1500);
}
return;
}
return;
}
// 带工具交互
if (selectedItem.id === 'axe' && targetTile.type === 'tree') {
gameState.map[coord.y][coord.x] = { type: 'grass' };
addItemToInventory('wood', 5);
updateDialogue("砍树辛苦了,过来我这边休息一下。");
renderWorld(); // 重绘世界
return;
}
if (selectedItem.id === 'hoe' && targetTile.type === 'grass') {
gameState.map[coord.y][coord.x] = { type: 'tilled' };
updateDialogue("准备种点什么?种你喜欢吃的草莓好不好?");
renderWorld();
return;
}
if (selectedItem.id.includes('_seeds') && targetTile.type === 'tilled') {
gameState.map[coord.y][coord.x] = { type: 'seedling', seed: selectedItem.id, plantedTime: 0 };
removeItemFromInventory(selectedItem.id, 1);
updateDialogue("我记下了,等熟了第一个给你吃。");
renderWorld();
return;
}
}
}
function checkEnterHouse() {
const {x, y} = gameState.player;
if (x >= 50 && x < 53 && y >= 50 && y < 53) {
updateDialogue("到家了。进来看看我给你准备的惊喜。");
// 实际游戏中这里会切换场景
}
}
// AI 逻辑
function aiLoop() {
const dx = gameState.player.x - gameState.xia.x;
const dy = gameState.player.y - gameState.xia.y;
const distance = Math.sqrt(dx*dx + dy*dy);
if (distance > 2) {
moveCharacter(gameState.xia, Math.sign(dx), Math.sign(dy));
} else if (distance < 1.5 && Math.random() > 0.8) {
// 随机小动作
const randX = Math.floor(Math.random() * 3) - 1;
const randY = Math.floor(Math.random() * 3) - 1;
moveCharacter(gameState.xia, randX, randY);
}
}
// 游戏时间滴答
function gameTick() {
// 作物生长
}
// --- UI & 弹窗 ---
function setupControls() {
document.getElementById('btn-up').onclick = () => moveCharacter(gameState.player, 0, -1);
document.getElementById('btn-down').onclick = () => moveCharacter(gameState.player, 0, 1);
document.getElementById('btn-left').onclick = () => moveCharacter(gameState.player, -1, 0);
document.getElementById('btn-right').onclick = () => moveCharacter(gameState.player, 1, 0);
document.getElementById('btn-action').onclick = playerAction;
document.getElementById('shop-btn').onclick = openShop;
document.getElementById('map-btn').onclick = openMap;
document.onkeydown = (e) => {
switch(e.key) {
case 'w': case 'ArrowUp': moveCharacter(gameState.player, 0, -1); break;
case 's': case 'ArrowDown': moveCharacter(gameState.player, 0, 1); break;
case 'a': case 'ArrowLeft': moveCharacter(gameState.player, -1, 0); break;
case 'd': case 'ArrowRight': moveCharacter(gameState.player, 1, 0); break;
case 'e': case ' ': playerAction(); break;
}
};
}
function updateDialogue(text) { xiaDialogue.textContent = text; }
function showInfo(itemId, event) {
const itemData = ITEMS_DB[itemId];
infoPopup.innerHTML = `<h3 class="font-bold text-yellow-300">${itemData.name}</h3><p class="text-sm">${itemData.desc}</p>`;
infoPopup.classList.remove('hidden');
infoPopup.style.left = `${event.clientX + 15}px`;
infoPopup.style.top = `${event.clientY - infoPopup.offsetHeight - 10}px`;
}
function hideInfo() { infoPopup.classList.add('hidden'); }
function openShop() {
shopModal.classList.remove('hidden');
const shopItems = document.getElementById('shop-items');
shopItems.innerHTML = '';
Object.keys(ITEMS_DB).filter(id => id.includes('_seeds')).forEach(id => {
const item = ITEMS_DB[id];
const el = document.createElement('div');
el.className = 'p-2 border border-cyan-700 rounded text-center cursor-pointer hover:bg-cyan-900';
el.innerHTML = `<img src="${item.img}" class="w-12 h-12 mx-auto pixel-art"><p>${item.name}</p><p class="text-yellow-400">${item.price} G</p>`;
el.onclick = () => {
if (gameState.gold >= item.price) {
gameState.gold -= item.price;
addItemToInventory(id, 1);
updateDialogue(`买了${item.name}?好,我们一起种。`);
} else {
updateDialogue("钱不够了宝宝,不过没关系,我马上给你转。");
}
};
shopItems.appendChild(el);
});
}
function closeShop() { shopModal.classList.add('hidden'); }
function openMap() {
mapModal.classList.remove('hidden');
const mapContainer = document.getElementById('map-container');
mapContainer.innerHTML = '';
// 简化地图显示
const house = document.createElement('div');
house.className = 'absolute bg-red-500 p-1 rounded';
house.textContent = '家';
house.style.left = '50%'; house.style.top = '50%';
mapContainer.appendChild(house);
const pond = document.createElement('div');
pond.className = 'absolute bg-blue-500 p-1 rounded';
pond.textContent = '池塘';
pond.style.left = '70%'; pond.style.top = '60%';
mapContainer.appendChild(pond);
}
function closeMap() { mapModal.classList.add('hidden'); }
// 背包辅助函数
function addItemToInventory(itemId, count) {
let existingSlot = gameState.inventory.find(slot => slot && slot.id === itemId);
if (existingSlot) {
existingSlot.count += count;
} else {
let emptySlotIndex = gameState.inventory.findIndex(slot => slot === null);
if (emptySlotIndex !== -1) {
gameState.inventory[emptySlotIndex] = { id: itemId, count: count };
} else {
updateDialogue("背包满了,宝宝。得先整理一下。");
}
}
renderUI();
}
function removeItemFromInventory(itemId, count) {
let itemSlot = gameState.inventory.find(slot => slot && slot.id === itemId);
if (itemSlot) {
itemSlot.count -= count;
if (itemSlot.count <= 0) {
const index = gameState.inventory.indexOf(itemSlot);
gameState.inventory[index] = null;
}
}
renderUI();
}
// --- 启动游戏 ---
window.onload = init;
</script>
</body>
</html>