Skip to content

Commit 612147c

Browse files
authored
drag and drop url fixes (microsoft#244671)
* fix url web stuffs * better logging * add comment * change to logservice don't use console. it is bad
1 parent b25957b commit 612147c

File tree

1 file changed

+58
-36
lines changed

1 file changed

+58
-36
lines changed

src/vs/workbench/contrib/chat/browser/chatDragAndDrop.ts

+58-36
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,8 @@ import { localize } from '../../../../nls.js';
2121
import { IDialogService } from '../../../../platform/dialogs/common/dialogs.js';
2222
import { CodeDataTransfers, containsDragType, DocumentSymbolTransferData, extractEditorsDropData, extractMarkerDropData, extractSymbolDropData, IDraggedResourceEditorInput, MarkerTransferData } from '../../../../platform/dnd/browser/dnd.js';
2323
import { IFileService } from '../../../../platform/files/common/files.js';
24+
import { ILogService } from '../../../../platform/log/common/log.js';
2425
import { MarkerSeverity } from '../../../../platform/markers/common/markers.js';
25-
import { INotificationService } from '../../../../platform/notification/common/notification.js';
2626
import { IThemeService, Themable } from '../../../../platform/theme/common/themeService.js';
2727
import { ISharedWebContentExtractorService } from '../../../../platform/webContentExtractor/common/webContentExtractor.js';
2828
import { isUntitledResourceEditorInput } from '../../../common/editor.js';
@@ -31,6 +31,7 @@ import { IEditorService } from '../../../services/editor/common/editorService.js
3131
import { IExtensionService, isProposedApiEnabled } from '../../../services/extensions/common/extensions.js';
3232
import { UntitledTextEditorInput } from '../../../services/untitled/common/untitledTextEditorInput.js';
3333
import { IChatRequestVariableEntry, IDiagnosticVariableEntry, IDiagnosticVariableEntryFilterData, ISymbolVariableEntry } from '../common/chatModel.js';
34+
import { IChatWidgetService } from './chat.js';
3435
import { ChatAttachmentModel } from './chatAttachmentModel.js';
3536
import { IChatInputStyles } from './chatInputPart.js';
3637
import { imageToHash } from './chatPasteProviders.js';
@@ -61,8 +62,9 @@ export class ChatDragAndDrop extends Themable {
6162
@IEditorService private readonly editorService: IEditorService,
6263
@IDialogService private readonly dialogService: IDialogService,
6364
@ITextModelService private readonly textModelService: ITextModelService,
64-
@INotificationService private readonly notificationService: INotificationService,
6565
@ISharedWebContentExtractorService private readonly webContentExtractorService: ISharedWebContentExtractorService,
66+
@IChatWidgetService private readonly chatWidgetService: IChatWidgetService,
67+
@ILogService private readonly logService: ILogService,
6668
) {
6769
super(themeService);
6870

@@ -332,10 +334,16 @@ export class ChatDragAndDrop extends Themable {
332334
return extractedImages.buffer;
333335
}
334336
} catch (error) {
335-
console.warn('Fetch failed:', error);
337+
this.logService.warn('Fetch failed:', error);
336338
}
337339

338-
this.notificationService.error(localize('fetchFailed', 'Failed to fetch image from URL: {0}', url));
340+
// TODO: use dnd provider to insert text @justschen
341+
const selection = this.chatWidgetService.lastFocusedWidget?.inputEditor.getSelection();
342+
if (selection && this.chatWidgetService.lastFocusedWidget) {
343+
this.chatWidgetService.lastFocusedWidget.inputEditor.executeEdits('chatInsertUrl', [{ range: selection, text: url }]);
344+
}
345+
346+
this.logService.warn(`Image URLs must end in .jpg, .png, .gif, .webp, or .bmp. Failed to fetch image from this URL: ${url}`);
339347
return undefined;
340348
}
341349

@@ -347,21 +355,21 @@ export class ChatDragAndDrop extends Themable {
347355
finalDisplayName = `${displayName} ${appendValue}`;
348356
}
349357

350-
const dataFromFile = await extractImageFromFile(e);
358+
const dataFromFile = await this.extractImageFromFile(e);
351359
if (dataFromFile) {
352360
return [await this.createImageVariable(await resizeImage(dataFromFile), finalDisplayName)];
353361
}
354362

355-
const dataFromUrl = await extractImageFromUrl(e);
363+
const dataFromUrl = await this.extractImageFromUrl(e);
356364
const variableEntries: IChatRequestVariableEntry[] = [];
357365
if (dataFromUrl) {
358366
for (const url of dataFromUrl) {
359367
if (/^data:image\/[a-z]+;base64,/.test(url)) {
360-
variableEntries.push(await this.createImageVariable(await resizeImage(url), finalDisplayName));
368+
variableEntries.push(await this.createImageVariable(await resizeImage(url), finalDisplayName, URI.parse(url)));
361369
} else if (/^https?:\/\/.+/.test(url)) {
362370
const imageData = await this.downloadImageAsUint8Array(url);
363371
if (imageData) {
364-
variableEntries.push(await this.createImageVariable(await resizeImage(imageData), finalDisplayName, url));
372+
variableEntries.push(await this.createImageVariable(await resizeImage(imageData), finalDisplayName, URI.parse(url), url));
365373
}
366374
}
367375
}
@@ -370,14 +378,15 @@ export class ChatDragAndDrop extends Themable {
370378
return variableEntries;
371379
}
372380

373-
private async createImageVariable(data: Uint8Array, name: string, id?: string) {
381+
private async createImageVariable(data: Uint8Array, name: string, uri?: URI, id?: string,): Promise<IChatRequestVariableEntry> {
374382
return {
375383
id: id || await imageToHash(data),
376384
name: name,
377385
value: data,
378386
isImage: true,
379387
isFile: false,
380-
isDirectory: false
388+
isDirectory: false,
389+
references: uri ? [{ reference: uri, kind: 'reference' }] : []
381390
};
382391
}
383392

@@ -433,6 +442,45 @@ export class ChatDragAndDrop extends Themable {
433442
this.overlays.forEach(overlay => this.updateOverlayStyles(overlay.overlay));
434443
this.overlayTextBackground = this.getColor(this.styles.listBackground) || '';
435444
}
445+
446+
447+
448+
private async extractImageFromFile(e: DragEvent): Promise<Uint8Array | undefined> {
449+
const files = e.dataTransfer?.files;
450+
if (files && files.length > 0) {
451+
const file = files[0];
452+
if (file.type.startsWith('image/')) {
453+
try {
454+
const buffer = await file.arrayBuffer();
455+
return new Uint8Array(buffer);
456+
} catch (error) {
457+
this.logService.error('Error reading file:', error);
458+
return undefined;
459+
}
460+
}
461+
}
462+
463+
return undefined;
464+
}
465+
466+
private async extractImageFromUrl(e: DragEvent): Promise<string[] | undefined> {
467+
const textUrl = e.dataTransfer?.getData('text/uri-list');
468+
if (textUrl) {
469+
try {
470+
const uris = UriList.parse(textUrl);
471+
if (uris.length > 0) {
472+
return uris;
473+
}
474+
} catch (error) {
475+
this.logService.error('Error parsing URI list:', error);
476+
return undefined;
477+
}
478+
}
479+
480+
return undefined;
481+
}
482+
483+
436484
}
437485

438486
async function getResourceAttachContext(resource: URI, isDirectory: boolean, textModelService: ITextModelService): Promise<IChatRequestVariableEntry | undefined> {
@@ -499,29 +547,3 @@ function symbolId(resource: URI, range?: IRange): string {
499547
}
500548
return resource.fsPath + rangePart;
501549
}
502-
503-
async function extractImageFromFile(e: DragEvent): Promise<Uint8Array | undefined> {
504-
const files = e.dataTransfer?.files;
505-
if (files && files.length > 0) {
506-
const file = files[0];
507-
if (file.type.startsWith('image/')) {
508-
const dataTransferFiles = await file.bytes();
509-
return dataTransferFiles;
510-
}
511-
}
512-
513-
return undefined;
514-
}
515-
516-
async function extractImageFromUrl(e: DragEvent): Promise<string[] | undefined> {
517-
const textUrl = e.dataTransfer?.getData('text/uri-list');
518-
if (textUrl) {
519-
const uris = UriList.parse(textUrl);
520-
if (uris.length > 0) {
521-
return uris;
522-
}
523-
}
524-
525-
return undefined;
526-
}
527-

0 commit comments

Comments
 (0)