cocos2d-cli 1.5.2 → 1.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,265 @@
1
+ /**
2
+ * Screenshot Core Module
3
+ * 渲染 JSON 数据并使用 Playwright 截图
4
+ */
5
+
6
+ const { chromium } = require('playwright');
7
+ const fs = require('fs').promises;
8
+ const path = require('path');
9
+ const http = require('http');
10
+ const url = require('url');
11
+ const os = require('os');
12
+
13
+ // 默认配置
14
+ const DEFAULT_CONFIG = {
15
+ jsonPath: null,
16
+ outputDir: process.cwd(),
17
+ viewport: { width: 750, height: 1334 },
18
+ fullPage: true,
19
+ debugBounds: false,
20
+ timeout: 30000,
21
+ waitTime: 1000
22
+ };
23
+
24
+ // 获取内置资源目录
25
+ function getAssetsDir() {
26
+ return path.join(__dirname, 'screenshot');
27
+ }
28
+
29
+ // 创建临时工作目录
30
+ async function createTempWorkDir() {
31
+ const tempBase = os.tmpdir();
32
+ const timestamp = Date.now();
33
+ const tempDir = path.join(tempBase, `cocos2d-screenshot-${timestamp}`);
34
+ await fs.mkdir(tempDir, { recursive: true });
35
+ return tempDir;
36
+ }
37
+
38
+ // 复制内置资源到临时目录
39
+ async function copyBuiltInAssets(tempDir) {
40
+ const assetsDir = getAssetsDir();
41
+ const assets = ['index.html', 'favicon.ico'];
42
+
43
+ for (const asset of assets) {
44
+ const src = path.join(assetsDir, asset);
45
+ const dest = path.join(tempDir, asset);
46
+ try {
47
+ await fs.copyFile(src, dest);
48
+ } catch (err) {
49
+ console.log(`Warning: Could not copy ${asset}: ${err.message}`);
50
+ }
51
+ }
52
+ }
53
+
54
+ // 静态文件服务器
55
+ async function startServer(staticDir) {
56
+ return new Promise((resolve, reject) => {
57
+ const server = http.createServer(async (req, res) => {
58
+ try {
59
+ const parsedUrl = url.parse(req.url);
60
+ let filePath = path.join(staticDir, parsedUrl.pathname);
61
+
62
+ if (parsedUrl.pathname === '/') {
63
+ filePath = path.join(staticDir, 'index.html');
64
+ }
65
+
66
+ const data = await fs.readFile(filePath);
67
+
68
+ const ext = path.extname(filePath).toLowerCase();
69
+ const contentTypes = {
70
+ '.html': 'text/html',
71
+ '.htm': 'text/html',
72
+ '.json': 'application/json',
73
+ '.js': 'application/javascript',
74
+ '.mjs': 'application/javascript',
75
+ '.css': 'text/css',
76
+ '.png': 'image/png',
77
+ '.jpg': 'image/jpeg',
78
+ '.jpeg': 'image/jpeg',
79
+ '.gif': 'image/gif',
80
+ '.svg': 'image/svg+xml',
81
+ '.ico': 'image/x-icon',
82
+ '.txt': 'text/plain',
83
+ '.xml': 'application/xml'
84
+ };
85
+
86
+ res.writeHead(200, {
87
+ 'Content-Type': contentTypes[ext] || 'application/octet-stream',
88
+ 'Access-Control-Allow-Origin': '*'
89
+ });
90
+ res.end(data);
91
+
92
+ } catch (err) {
93
+ if (err.code === 'ENOENT') {
94
+ res.writeHead(404);
95
+ res.end();
96
+ } else {
97
+ console.error(`[Server Error] ${req.url}:`, err.message);
98
+ res.writeHead(500);
99
+ res.end();
100
+ }
101
+ }
102
+ });
103
+
104
+ server.listen(0, '127.0.0.1', () => resolve(server));
105
+ });
106
+ }
107
+
108
+ // 递归删除目录
109
+ async function removeDir(dirPath) {
110
+ try {
111
+ await fs.rm(dirPath, { recursive: true, force: true });
112
+ } catch (err) {
113
+ console.log(`Warning: Could not remove temp dir: ${err.message}`);
114
+ }
115
+ }
116
+
117
+ /**
118
+ * 截图核心函数
119
+ * @param {Object} userConfig - 配置选项
120
+ * @param {string} userConfig.jsonPath - JSON 文件路径
121
+ * @param {string} userConfig.outputDir - 输出目录
122
+ * @param {Object} userConfig.viewport - 视口大小 {width, height}
123
+ * @param {boolean} userConfig.fullPage - 是否全页截图
124
+ * @param {number} userConfig.timeout - 超时时间(毫秒)
125
+ * @param {number} userConfig.waitTime - 等待时间(毫秒)
126
+ * @returns {Promise<{screenshotPath: string, logs: Array}>}
127
+ */
128
+ async function takeScreenshot(userConfig = {}) {
129
+ const config = { ...DEFAULT_CONFIG, ...userConfig };
130
+
131
+ if (!config.jsonPath) {
132
+ throw new Error('JSON file path is required');
133
+ }
134
+
135
+ let server = null;
136
+ let browser = null;
137
+ let tempDir = null;
138
+ const logs = [];
139
+ let screenshotPath = null;
140
+
141
+ try {
142
+ const timestamp = Date.now();
143
+
144
+ // 检查 JSON 文件是否存在
145
+ try {
146
+ await fs.access(config.jsonPath);
147
+ } catch (error) {
148
+ throw new Error(`JSON file not found: ${config.jsonPath}`);
149
+ }
150
+
151
+ // 确保输出目录存在
152
+ await fs.mkdir(config.outputDir, { recursive: true });
153
+
154
+ // 创建临时工作目录
155
+ tempDir = await createTempWorkDir();
156
+ console.log(`Temp dir: ${tempDir}`);
157
+
158
+ // 复制内置资源到临时目录
159
+ await copyBuiltInAssets(tempDir);
160
+
161
+ // 复制用户的 JSON 文件到临时目录
162
+ const destJsonPath = path.join(tempDir, 'data.json');
163
+ await fs.copyFile(config.jsonPath, destJsonPath);
164
+ console.log(`JSON: ${config.jsonPath}`);
165
+
166
+ // 截图输出路径
167
+ screenshotPath = path.join(config.outputDir, `screenshot-${timestamp}.png`);
168
+
169
+ // 启动HTTP服务器
170
+ console.log('\n=== Starting Server ===');
171
+ server = await startServer(tempDir);
172
+ const serverUrl = `http://127.0.0.1:${server.address().port}`;
173
+ console.log(`Server: ${serverUrl}`);
174
+
175
+ // 启动浏览器
176
+ console.log('\n=== Launching Browser ===');
177
+ browser = await chromium.launch({
178
+ headless: true,
179
+ channel: 'chrome'
180
+ });
181
+ console.log('Browser launched');
182
+
183
+ const page = await browser.newPage({
184
+ viewport: config.viewport
185
+ });
186
+
187
+ // 监听浏览器控制台日志
188
+ page.on('console', msg => {
189
+ const text = msg.text();
190
+ logs.push({
191
+ timestamp: new Date().toISOString(),
192
+ type: msg.type(),
193
+ text
194
+ });
195
+
196
+ if (msg.type() === 'error') {
197
+ console.error(`[Browser Error] ${text}`);
198
+ } else if (msg.type() === 'warning') {
199
+ console.warn(`[Browser Warning] ${text}`);
200
+ } else {
201
+ console.log(`[Browser] ${text}`);
202
+ }
203
+ });
204
+
205
+ page.on('pageerror', error => {
206
+ logs.push({
207
+ timestamp: new Date().toISOString(),
208
+ type: 'pageerror',
209
+ text: error.message
210
+ });
211
+ console.error('[Page Error]', error.message);
212
+ });
213
+
214
+ page.on('requestfailed', request => {
215
+ const failure = request.failure();
216
+ const errorText = failure ? failure.errorText : 'Unknown error';
217
+ logs.push({
218
+ timestamp: new Date().toISOString(),
219
+ type: 'requestfailed',
220
+ url: request.url(),
221
+ error: errorText
222
+ });
223
+ console.error(`[Request Failed] ${request.url()}`);
224
+ });
225
+
226
+ // 加载页面
227
+ console.log('\n=== Loading Page ===');
228
+ const pageUrl = config.debugBounds
229
+ ? `${serverUrl}/index.html?debugBounds=true`
230
+ : `${serverUrl}/index.html`;
231
+ await page.goto(pageUrl, {
232
+ waitUntil: 'networkidle',
233
+ timeout: config.timeout
234
+ });
235
+ console.log('Page loaded');
236
+
237
+ // 等待渲染
238
+ console.log('\n=== Waiting for Render ===');
239
+ await page.waitForTimeout(config.waitTime);
240
+ console.log('Wait complete');
241
+
242
+ // 截图
243
+ console.log('\n=== Taking Screenshot ===');
244
+ await page.screenshot({
245
+ path: screenshotPath,
246
+ fullPage: config.fullPage
247
+ });
248
+ console.log(`Screenshot saved: ${screenshotPath}`);
249
+
250
+ console.log('\n=== Done ===');
251
+
252
+ return { screenshotPath, logs };
253
+
254
+ } catch (error) {
255
+ console.error('\nError:', error.message);
256
+ throw error;
257
+ } finally {
258
+ // 清理资源
259
+ if (browser) await browser.close().catch(() => {});
260
+ if (server) server.close();
261
+ if (tempDir) await removeDir(tempDir);
262
+ }
263
+ }
264
+
265
+ module.exports = { takeScreenshot };