486CHD commited on
Commit
96a62d2
·
verified ·
1 Parent(s): c581231

Upload 13 files

Browse files
Files changed (2) hide show
  1. index.html +9 -28
  2. server.js +110 -38
index.html CHANGED
@@ -1653,8 +1653,8 @@
1653
  if (queue.length === 0) return;
1654
 
1655
  this.thumbProcessing = true;
1656
- const maxWidth = Device.isMobile ? 360 : 768;
1657
- const quality = Device.isMobile ? 0.6 : 0.76;
1658
  const processNext = async () => {
1659
  const item = queue.shift();
1660
  if (!item) {
@@ -1667,14 +1667,7 @@
1667
  return;
1668
  }
1669
  const source = item.image || item.imageUrl;
1670
- let sourceForThumb = source;
1671
- if (Device.isMobile && sourceForThumb && sourceForThumb.startsWith('data:image') && sourceForThumb.length > 900000) {
1672
- const reduced = await ImageHandler.reduceImageForMobile(sourceForThumb);
1673
- if (reduced) {
1674
- sourceForThumb = reduced;
1675
- }
1676
- }
1677
- const thumb = await ImageHandler.createThumbnail(sourceForThumb, maxWidth, quality);
1678
  if (thumb && thumb !== source) {
1679
  item.thumb = thumb;
1680
  Database.update(item.id, { thumb }).catch(() => {});
@@ -2390,8 +2383,8 @@
2390
  if (queue.length === 0) return;
2391
 
2392
  this.thumbProcessing = true;
2393
- const maxWidth = 360;
2394
- const quality = 0.6;
2395
  const processNext = async () => {
2396
  const item = queue.shift();
2397
  if (!item) {
@@ -2404,14 +2397,7 @@
2404
  return;
2405
  }
2406
  const source = item.image || item.imageUrl;
2407
- let sourceForThumb = source;
2408
- if (sourceForThumb && sourceForThumb.startsWith('data:image') && sourceForThumb.length > 900000) {
2409
- const reduced = await ImageHandler.reduceImageForMobile(sourceForThumb);
2410
- if (reduced) {
2411
- sourceForThumb = reduced;
2412
- }
2413
- }
2414
- const thumb = await ImageHandler.createThumbnail(sourceForThumb, maxWidth, quality);
2415
  if (thumb && thumb !== source) {
2416
  item.thumb = thumb;
2417
  }
@@ -2960,7 +2946,7 @@
2960
  prompt: prompt,
2961
  images: AppState.currentImages,
2962
  preferUrl: false,
2963
- imageSize: Device.isMobile ? '1K' : '4K'
2964
  };
2965
 
2966
  const response = await apiFetch('/api/generate', {
@@ -3025,14 +3011,9 @@
3025
  if (Device.isMobile && previewImage.startsWith('data:image') && previewImage.length > 900000) {
3026
  const reduceTask = async () => {
3027
  try {
3028
- let sourceForThumb = previewImage;
3029
- const reduced = await ImageHandler.reduceImageForMobile(sourceForThumb);
3030
- if (reduced) {
3031
- sourceForThumb = reduced;
3032
- }
3033
  if (!newItem.thumb) {
3034
- const newThumb = await ImageHandler.createThumbnail(sourceForThumb, 360, 0.6);
3035
- if (newThumb && newThumb !== sourceForThumb) {
3036
  newItem.thumb = newThumb;
3037
  }
3038
  }
 
1653
  if (queue.length === 0) return;
1654
 
1655
  this.thumbProcessing = true;
1656
+ const maxWidth = Device.isMobile ? 640 : 768;
1657
+ const quality = Device.isMobile ? 0.75 : 0.76;
1658
  const processNext = async () => {
1659
  const item = queue.shift();
1660
  if (!item) {
 
1667
  return;
1668
  }
1669
  const source = item.image || item.imageUrl;
1670
+ const thumb = await ImageHandler.createThumbnail(source, maxWidth, quality);
 
 
 
 
 
 
 
1671
  if (thumb && thumb !== source) {
1672
  item.thumb = thumb;
1673
  Database.update(item.id, { thumb }).catch(() => {});
 
2383
  if (queue.length === 0) return;
2384
 
2385
  this.thumbProcessing = true;
2386
+ const maxWidth = 640;
2387
+ const quality = 0.75;
2388
  const processNext = async () => {
2389
  const item = queue.shift();
2390
  if (!item) {
 
2397
  return;
2398
  }
2399
  const source = item.image || item.imageUrl;
2400
+ const thumb = await ImageHandler.createThumbnail(source, maxWidth, quality);
 
 
 
 
 
 
 
2401
  if (thumb && thumb !== source) {
2402
  item.thumb = thumb;
2403
  }
 
2946
  prompt: prompt,
2947
  images: AppState.currentImages,
2948
  preferUrl: false,
2949
+ imageSize: '4K'
2950
  };
2951
 
2952
  const response = await apiFetch('/api/generate', {
 
3011
  if (Device.isMobile && previewImage.startsWith('data:image') && previewImage.length > 900000) {
3012
  const reduceTask = async () => {
3013
  try {
 
 
 
 
 
3014
  if (!newItem.thumb) {
3015
+ const newThumb = await ImageHandler.createThumbnail(previewImage, 640, 0.75);
3016
+ if (newThumb && newThumb !== previewImage) {
3017
  newItem.thumb = newThumb;
3018
  }
3019
  }
server.js CHANGED
@@ -87,12 +87,12 @@ const authMiddleware = (req, res, next) => {
87
  // ============================================
88
  // 图片数据解析模块
89
  // ============================================
90
- const ImageParser = {
91
  /**
92
  * 从 assistant content 中提取 base64 图片数据
93
  * 格式: ![Generated Image](data:image/png;base64,xxxxx)
94
  */
95
- extractBase64FromMarkdown(content) {
96
  if (!content || typeof content !== 'string') {
97
  return null;
98
  }
@@ -116,13 +116,30 @@ const ImageParser = {
116
  }
117
 
118
  return null;
119
- },
120
-
121
- isValidBase64Image(base64Data) {
122
- if (!base64Data) return false;
123
- return base64Data.startsWith('data:image/');
124
- }
125
- };
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
126
 
127
  // ============================================
128
  // 消息构建模块
@@ -214,7 +231,7 @@ const APIService = {
214
  /**
215
  * Extract image from Gemini response
216
  */
217
- extractImageFromResponse(data) {
218
  console.log(`[${new Date().toISOString()}] API response type:`, typeof data);
219
  console.log(`[${new Date().toISOString()}] API response keys:`, Object.keys(data || {}));
220
 
@@ -225,34 +242,89 @@ const APIService = {
225
  throw new Error(`API returned error: ${message}`);
226
  }
227
 
228
- const candidates = Array.isArray(data.candidates) ? data.candidates : [];
229
- for (const candidate of candidates) {
230
- const parts = candidate?.content?.parts || [];
231
- for (const part of parts) {
232
- const inlineData = part.inlineData || part.inline_data;
233
- if (inlineData && inlineData.data) {
234
- const mimeType = inlineData.mimeType || inlineData.mime_type || 'image/png';
235
- return `data:${mimeType};base64,${inlineData.data}`;
236
- }
237
- if (part.text) {
238
- const imageData = ImageParser.extractBase64FromMarkdown(part.text);
239
- if (imageData && ImageParser.isValidBase64Image(imageData)) {
240
- return imageData;
241
- }
242
- }
243
- }
244
- }
245
-
246
- if (data.data && data.data[0]) {
247
- if (data.data[0].b64_json) {
248
- console.log(`[${new Date().toISOString()}] Using DALL-E b64_json format`);
249
- return `data:image/png;base64,${data.data[0].b64_json}`;
250
- }
251
- if (data.data[0].url) {
252
- console.log(`[${new Date().toISOString()}] Using DALL-E URL format`);
253
- return data.data[0].url;
254
- }
255
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
256
 
257
  console.log(`[${new Date().toISOString()}] Full API response:`, JSON.stringify(data, null, 2));
258
  throw new Error('Unable to extract image data from API response.');
 
87
  // ============================================
88
  // 图片数据解析模块
89
  // ============================================
90
+ const ImageParser = {
91
  /**
92
  * 从 assistant content 中提取 base64 图片数据
93
  * 格式: ![Generated Image](data:image/png;base64,xxxxx)
94
  */
95
+ extractBase64FromMarkdown(content) {
96
  if (!content || typeof content !== 'string') {
97
  return null;
98
  }
 
116
  }
117
 
118
  return null;
119
+ },
120
+
121
+ normalizeBase64Image(value) {
122
+ if (!value || typeof value !== 'string') return null;
123
+ const trimmed = value.trim();
124
+ if (!trimmed) return null;
125
+ if (trimmed.startsWith('data:image/')) {
126
+ return trimmed;
127
+ }
128
+ if (/^https?:\/\//i.test(trimmed)) {
129
+ return trimmed;
130
+ }
131
+ const cleaned = trimmed.replace(/\s+/g, '');
132
+ if (/^[A-Za-z0-9+/=_-]+$/.test(cleaned) && cleaned.length > 2000) {
133
+ return `data:image/png;base64,${cleaned}`;
134
+ }
135
+ return null;
136
+ },
137
+
138
+ isValidBase64Image(base64Data) {
139
+ if (!base64Data) return false;
140
+ return base64Data.startsWith('data:image/');
141
+ }
142
+ };
143
 
144
  // ============================================
145
  // 消息构建模块
 
231
  /**
232
  * Extract image from Gemini response
233
  */
234
+ extractImageFromResponse(data) {
235
  console.log(`[${new Date().toISOString()}] API response type:`, typeof data);
236
  console.log(`[${new Date().toISOString()}] API response keys:`, Object.keys(data || {}));
237
 
 
242
  throw new Error(`API returned error: ${message}`);
243
  }
244
 
245
+ const candidates = Array.isArray(data.candidates) ? data.candidates : [];
246
+ for (const candidate of candidates) {
247
+ const parts = candidate?.content?.parts || [];
248
+ for (const part of parts) {
249
+ const inlineData = part.inlineData || part.inline_data;
250
+ if (inlineData && inlineData.data) {
251
+ const mimeType = inlineData.mimeType || inlineData.mime_type || 'image/png';
252
+ return `data:${mimeType};base64,${inlineData.data}`;
253
+ }
254
+ if (part.text) {
255
+ const imageData = ImageParser.extractBase64FromMarkdown(part.text);
256
+ if (imageData && ImageParser.isValidBase64Image(imageData)) {
257
+ return imageData;
258
+ }
259
+ const normalizedText = ImageParser.normalizeBase64Image(part.text);
260
+ if (normalizedText) {
261
+ return normalizedText;
262
+ }
263
+ }
264
+ }
265
+ }
266
+
267
+ const extractFromValue = (value) => {
268
+ if (!value) return null;
269
+ if (typeof value === 'string') {
270
+ return ImageParser.normalizeBase64Image(value);
271
+ }
272
+ if (typeof value === 'object') {
273
+ if (value.data) {
274
+ const mimeType = value.mimeType || value.mime_type || 'image/png';
275
+ return `data:${mimeType};base64,${value.data}`;
276
+ }
277
+ if (value.b64_json) {
278
+ return `data:image/png;base64,${value.b64_json}`;
279
+ }
280
+ if (value.url) {
281
+ return value.url;
282
+ }
283
+ }
284
+ return null;
285
+ };
286
+
287
+ const directCandidates = [
288
+ data.image,
289
+ data.image_base64,
290
+ data.imageBase64,
291
+ data.base64,
292
+ data.b64_json,
293
+ data.result,
294
+ data.output
295
+ ];
296
+ for (const entry of directCandidates) {
297
+ const extracted = extractFromValue(entry);
298
+ if (extracted) return extracted;
299
+ }
300
+
301
+ if (Array.isArray(data.images)) {
302
+ for (const entry of data.images) {
303
+ const extracted = extractFromValue(entry);
304
+ if (extracted) return extracted;
305
+ }
306
+ }
307
+
308
+ if (Array.isArray(data.output)) {
309
+ for (const output of data.output) {
310
+ const content = Array.isArray(output?.content) ? output.content : [];
311
+ for (const part of content) {
312
+ const extracted = extractFromValue(part?.image || part?.image_url || part?.b64_json || part);
313
+ if (extracted) return extracted;
314
+ }
315
+ }
316
+ }
317
+
318
+ if (data.data && data.data[0]) {
319
+ if (data.data[0].b64_json) {
320
+ console.log(`[${new Date().toISOString()}] Using DALL-E b64_json format`);
321
+ return `data:image/png;base64,${data.data[0].b64_json}`;
322
+ }
323
+ if (data.data[0].url) {
324
+ console.log(`[${new Date().toISOString()}] Using DALL-E URL format`);
325
+ return data.data[0].url;
326
+ }
327
+ }
328
 
329
  console.log(`[${new Date().toISOString()}] Full API response:`, JSON.stringify(data, null, 2));
330
  throw new Error('Unable to extract image data from API response.');