Skip to content

Commit fe74180

Browse files
committed
.
1 parent 93a2654 commit fe74180

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

80 files changed

+5907
-6
lines changed

.gitmodules

+4-5
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,5 @@
1-
; packages
21

32

4-
; plugins
5-
63
[submodule "apps/exts_alfred_claude-search"]
74
path = apps/exts/alfred/claude-search
85
url = [email protected]:cs-magic/exts_alfred_claude-search.git
@@ -15,7 +12,6 @@
1512
path = apps/exts/chrome/claude-artifact-enhancer
1613
url = [email protected]:cs-magic/exts_chrome_claude-artifact-enhancer.git
1714

18-
; apps
1915
[submodule "apps/playground"]
2016
path = apps/playground
2117
url = [email protected]:cs-magic/playground.git
@@ -63,7 +59,7 @@
6359
path = apps/uni-pusher
6460
url = [email protected]:cs-magic/uni-pusher.git
6561

66-
; cases
62+
6763
[submodule "cases/zhijikeji"]
6864
path = cases/zhijikeji
6965
url = [email protected]:cs-magic/cases_zhijikeji.git
@@ -82,3 +78,6 @@
8278
[submodule "apps/docs"]
8379
path = apps/docs
8480
url = [email protected]:MarkShawn2020/nextra-docs.git
81+
[submodule "apps/exts/chrome/auto-sdk"]
82+
path = apps/exts/chrome/auto-sdk
83+
url = [email protected]:MarkShawn2020/auto-sdk.git

apps/badminton-reservation

Submodule badminton-reservation added at 0115320

apps/cs-magic/eval-ai

Submodule eval-ai added at 7650985

apps/cs-magic/poketto

Submodule poketto added at 00b1a79

apps/cs-magic/thinkow

Submodule thinkow added at 058367a

apps/docs

Submodule docs added at 332424d

apps/exts/alfred/claude-search

Submodule claude-search added at 26cf8e7

apps/exts/chrome/auto-sdk

Submodule auto-sdk added at e85c872

apps/exts/chrome/chatbot-exporter

Submodule chatbot-exporter added at 6b0a264

apps/game/fruit/game.js

+193
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,193 @@
1+
class FruitGame {
2+
constructor() {
3+
this.canvas = document.getElementById('game-canvas');
4+
this.ctx = this.canvas.getContext('2d');
5+
this.scoreElement = document.getElementById('score-value');
6+
this.score = 0;
7+
this.fruits = [];
8+
this.slices = [];
9+
this.lastTime = 0;
10+
this.fruitInterval = 1000;
11+
this.lastFruitTime = 0;
12+
this.gameState = 'playing'; // 'playing', 'gameover'
13+
this.lives = 3;
14+
15+
this.fruitTypes = ['apple', 'orange', 'watermelon', 'pineapple', 'bomb'];
16+
this.fruitImages = {};
17+
this.loadImages();
18+
19+
this.sliceSound = new Audio(this.sliceSoundData);
20+
this.bombSound = new Audio(this.bombSoundData);
21+
this.backgroundMusic = new Audio(this.backgroundMusicData);
22+
this.backgroundMusic.loop = true;
23+
24+
this.resizeCanvas();
25+
window.addEventListener('resize', () => this.resizeCanvas());
26+
this.canvas.addEventListener('touchstart', (e) => this.handleTouch(e));
27+
this.canvas.addEventListener('touchmove', (e) => this.handleTouch(e));
28+
29+
this.backgroundMusic.play();
30+
requestAnimationFrame((time) => this.gameLoop(time));
31+
}
32+
33+
loadImages() {
34+
const imageData = {
35+
apple: '',
36+
orange: '',
37+
watermelon: '',
38+
pineapple: '',
39+
bomb: ''
40+
};
41+
this.fruitImages = imageData;
42+
}
43+
44+
resizeCanvas() {
45+
this.canvas.width = window.innerWidth;
46+
this.canvas.height = window.innerHeight;
47+
}
48+
49+
handleTouch(e) {
50+
e.preventDefault();
51+
const touch = e.touches[0];
52+
const rect = this.canvas.getBoundingClientRect();
53+
const x = touch.clientX - rect.left;
54+
const y = touch.clientY - rect.top;
55+
this.checkFruitCut(x, y);
56+
this.slices.push({x, y, time: Date.now()});
57+
}
58+
59+
checkFruitCut(x, y) {
60+
this.fruits.forEach((fruit, index) => {
61+
if (this.isPointInFruit(x, y, fruit)) {
62+
if (fruit.type === 'bomb') {
63+
this.bombSound.play();
64+
this.lives--;
65+
if (this.lives <= 0) {
66+
this.gameState = 'gameover';
67+
}
68+
} else {
69+
this.sliceSound.play();
70+
this.fruits.splice(index, 1);
71+
this.score += fruit.points;
72+
this.scoreElement.textContent = this.score;
73+
this.createFruitSlices(fruit);
74+
}
75+
}
76+
});
77+
}
78+
79+
isPointInFruit(x, y, fruit) {
80+
const dx = x - fruit.x;
81+
const dy = y - fruit.y;
82+
return dx * dx + dy * dy <= fruit.radius * fruit.radius;
83+
}
84+
85+
createFruitSlices(fruit) {
86+
// Create two halves of the fruit
87+
for (let i = 0; i < 2; i++) {
88+
this.fruits.push({
89+
...fruit,
90+
isSlice: true,
91+
rotationSpeed: (Math.random() - 0.5) * 0.2,
92+
speedX: (Math.random() - 0.5) * 5
93+
});
94+
}
95+
}
96+
97+
spawnFruit() {
98+
const type = this.fruitTypes[Math.floor(Math.random() * this.fruitTypes.length)];
99+
const fruit = {
100+
x: Math.random() * this.canvas.width,
101+
y: this.canvas.height,
102+
radius: 30,
103+
speedY: -10 - Math.random() * 5,
104+
rotation: 0,
105+
rotationSpeed: (Math.random() - 0.5) * 0.2,
106+
type: type,
107+
points: type === 'bomb' ? 0 : 10
108+
};
109+
this.fruits.push(fruit);
110+
}
111+
112+
updateFruits(deltaTime) {
113+
this.fruits.forEach((fruit, index) => {
114+
fruit.y += fruit.speedY * deltaTime / 16;
115+
fruit.x += (fruit.speedX || 0) * deltaTime / 16;
116+
fruit.rotation += fruit.rotationSpeed * deltaTime / 16;
117+
fruit.speedY += 0.2; // gravity
118+
if (fruit.y - fruit.radius > this.canvas.height) {
119+
if (!fruit.isSlice && fruit.type !== 'bomb') {
120+
this.lives--;
121+
if (this.lives <= 0) {
122+
this.gameState = 'gameover';
123+
}
124+
}
125+
this.fruits.splice(index, 1);
126+
}
127+
});
128+
}
129+
130+
drawFruits() {
131+
this.fruits.forEach(fruit => {
132+
this.ctx.save();
133+
this.ctx.translate(fruit.x, fruit.y);
134+
this.ctx.rotate(fruit.rotation);
135+
this.ctx.drawImage(this.fruitImages[fruit.type], -fruit.radius, -fruit.radius, fruit.radius * 2, fruit.radius * 2);
136+
this.ctx.restore();
137+
});
138+
}
139+
140+
drawSlices() {
141+
this.ctx.strokeStyle = 'white';
142+
this.ctx.lineWidth = 3;
143+
this.ctx.beginPath();
144+
this.slices.forEach((slice, index) => {
145+
if (index === 0) {
146+
this.ctx.moveTo(slice.x, slice.y);
147+
} else {
148+
this.ctx.lineTo(slice.x, slice.y);
149+
}
150+
if (Date.now() - slice.time > 100) {
151+
this.slices.splice(index, 1);
152+
}
153+
});
154+
this.ctx.stroke();
155+
}
156+
157+
drawLives() {
158+
this.ctx.fillStyle = 'red';
159+
for (let i = 0; i < this.lives; i++) {
160+
this.ctx.fillRect(10 + i * 30, 40, 20, 20);
161+
}
162+
}
163+
164+
gameLoop(currentTime) {
165+
const deltaTime = currentTime - this.lastTime;
166+
this.lastTime = currentTime;
167+
168+
this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height);
169+
170+
if (this.gameState === 'playing') {
171+
if (currentTime - this.lastFruitTime > this.fruitInterval) {
172+
this.spawnFruit();
173+
this.lastFruitTime = currentTime;
174+
}
175+
176+
this.updateFruits(deltaTime);
177+
this.drawFruits();
178+
this.drawSlices();
179+
this.drawLives();
180+
} else if (this.gameState === 'gameover') {
181+
this.ctx.fillStyle = 'black';
182+
this.ctx.font = '48px Arial';
183+
this.ctx.fillText('Game Over', this.canvas.width / 2 - 100, this.canvas.height / 2);
184+
this.ctx.fillText(`Score: ${this.score}`, this.canvas.width / 2 - 70, this.canvas.height / 2 + 60);
185+
}
186+
187+
requestAnimationFrame((time) => this.gameLoop(time));
188+
}
189+
}
190+
191+
window.onload = () => {
192+
new FruitGame();
193+
};

apps/game/fruit/index.html

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<!DOCTYPE html>
2+
<html lang="zh-CN">
3+
<head>
4+
<meta charset="UTF-8">
5+
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no">
6+
<title>切水果游戏</title>
7+
<link rel="stylesheet" href="style.css">
8+
</head>
9+
<body>
10+
<div id="game-container">
11+
<canvas id="game-canvas"></canvas>
12+
<div id="score">分数: <span id="score-value">0</span></div>
13+
</div>
14+
<script src="game.js"></script>
15+
</body>
16+
</html>

apps/game/fruit/readme.md

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
# 手机网页端切水果游戏
2+
3+
这是一个简单而有趣的切水果游戏,专为手机网页端设计。玩家通过触摸屏幕来切割飞来的水果,获得分数。
4+
5+
## 游戏特点
6+
7+
- 响应式设计,适合各种手机屏幕
8+
- 流畅的动画效果
9+
- 逐渐增加的难度
10+
- 计分系统
11+
12+
## 技术栈
13+
14+
- HTML5
15+
- CSS3
16+
- JavaScript (使用 Canvas API)
17+
18+
## 开发计划
19+
20+
1. 创建基本的游戏界面
21+
2. 实现水果的生成和动画
22+
3. 添加切割检测和效果
23+
4. 实现计分系统
24+
5. 添加音效
25+
6. 优化性能和触摸响应
26+
27+
## 如何运行
28+
29+
1. 克隆此仓库
30+
2. 打开 `index.html` 文件在浏览器中运行
31+
3. 在手机浏览器中访问部署后的URL
32+
33+
## 贡献
34+
35+
欢迎提出建议和改进意见!
36+

apps/game/fruit/style.css

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
body, html {
2+
margin: 0;
3+
padding: 0;
4+
width: 100%;
5+
height: 100%;
6+
overflow: hidden;
7+
background-color: #87CEEB; /* 天蓝色背景 */
8+
}
9+
10+
#game-container {
11+
position: relative;
12+
width: 100%;
13+
height: 100%;
14+
}
15+
16+
#game-canvas {
17+
position: absolute;
18+
top: 0;
19+
left: 0;
20+
width: 100%;
21+
height: 100%;
22+
}
23+
24+
#score {
25+
position: absolute;
26+
top: 10px;
27+
left: 10px;
28+
font-size: 24px;
29+
color: white;
30+
text-shadow: 2px 2px 4px rgba(0,0,0,0.5);
31+
z-index: 10;
32+
}

0 commit comments

Comments
 (0)