Spaces:
Running
Running
Upload 13 files
Browse files- index.html +9 -28
- 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 ?
|
| 1657 |
-
const quality = Device.isMobile ? 0.
|
| 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 |
-
|
| 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 =
|
| 2394 |
-
const quality = 0.
|
| 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 |
-
|
| 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:
|
| 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(
|
| 3035 |
-
if (newThumb && newThumb !==
|
| 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 |
* 格式: 
|
| 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 |
-
|
| 122 |
-
if (!
|
| 123 |
-
|
| 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 |
-
|
| 247 |
-
|
| 248 |
-
|
| 249 |
-
|
| 250 |
-
|
| 251 |
-
if (
|
| 252 |
-
|
| 253 |
-
return
|
| 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 |
* 格式: 
|
| 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.');
|