@@ -21,8 +21,8 @@ import { localize } from '../../../../nls.js';
21
21
import { IDialogService } from '../../../../platform/dialogs/common/dialogs.js' ;
22
22
import { CodeDataTransfers , containsDragType , DocumentSymbolTransferData , extractEditorsDropData , extractMarkerDropData , extractSymbolDropData , IDraggedResourceEditorInput , MarkerTransferData } from '../../../../platform/dnd/browser/dnd.js' ;
23
23
import { IFileService } from '../../../../platform/files/common/files.js' ;
24
+ import { ILogService } from '../../../../platform/log/common/log.js' ;
24
25
import { MarkerSeverity } from '../../../../platform/markers/common/markers.js' ;
25
- import { INotificationService } from '../../../../platform/notification/common/notification.js' ;
26
26
import { IThemeService , Themable } from '../../../../platform/theme/common/themeService.js' ;
27
27
import { ISharedWebContentExtractorService } from '../../../../platform/webContentExtractor/common/webContentExtractor.js' ;
28
28
import { isUntitledResourceEditorInput } from '../../../common/editor.js' ;
@@ -31,6 +31,7 @@ import { IEditorService } from '../../../services/editor/common/editorService.js
31
31
import { IExtensionService , isProposedApiEnabled } from '../../../services/extensions/common/extensions.js' ;
32
32
import { UntitledTextEditorInput } from '../../../services/untitled/common/untitledTextEditorInput.js' ;
33
33
import { IChatRequestVariableEntry , IDiagnosticVariableEntry , IDiagnosticVariableEntryFilterData , ISymbolVariableEntry } from '../common/chatModel.js' ;
34
+ import { IChatWidgetService } from './chat.js' ;
34
35
import { ChatAttachmentModel } from './chatAttachmentModel.js' ;
35
36
import { IChatInputStyles } from './chatInputPart.js' ;
36
37
import { imageToHash } from './chatPasteProviders.js' ;
@@ -61,8 +62,9 @@ export class ChatDragAndDrop extends Themable {
61
62
@IEditorService private readonly editorService : IEditorService ,
62
63
@IDialogService private readonly dialogService : IDialogService ,
63
64
@ITextModelService private readonly textModelService : ITextModelService ,
64
- @INotificationService private readonly notificationService : INotificationService ,
65
65
@ISharedWebContentExtractorService private readonly webContentExtractorService : ISharedWebContentExtractorService ,
66
+ @IChatWidgetService private readonly chatWidgetService : IChatWidgetService ,
67
+ @ILogService private readonly logService : ILogService ,
66
68
) {
67
69
super ( themeService ) ;
68
70
@@ -332,10 +334,16 @@ export class ChatDragAndDrop extends Themable {
332
334
return extractedImages . buffer ;
333
335
}
334
336
} catch ( error ) {
335
- console . warn ( 'Fetch failed:' , error ) ;
337
+ this . logService . warn ( 'Fetch failed:' , error ) ;
336
338
}
337
339
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 } ` ) ;
339
347
return undefined ;
340
348
}
341
349
@@ -347,21 +355,21 @@ export class ChatDragAndDrop extends Themable {
347
355
finalDisplayName = `${ displayName } ${ appendValue } ` ;
348
356
}
349
357
350
- const dataFromFile = await extractImageFromFile ( e ) ;
358
+ const dataFromFile = await this . extractImageFromFile ( e ) ;
351
359
if ( dataFromFile ) {
352
360
return [ await this . createImageVariable ( await resizeImage ( dataFromFile ) , finalDisplayName ) ] ;
353
361
}
354
362
355
- const dataFromUrl = await extractImageFromUrl ( e ) ;
363
+ const dataFromUrl = await this . extractImageFromUrl ( e ) ;
356
364
const variableEntries : IChatRequestVariableEntry [ ] = [ ] ;
357
365
if ( dataFromUrl ) {
358
366
for ( const url of dataFromUrl ) {
359
367
if ( / ^ d a t a : i m a g e \/ [ a - z ] + ; b a s e 6 4 , / . 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 ) ) ) ;
361
369
} else if ( / ^ h t t p s ? : \/ \/ .+ / . test ( url ) ) {
362
370
const imageData = await this . downloadImageAsUint8Array ( url ) ;
363
371
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 ) ) ;
365
373
}
366
374
}
367
375
}
@@ -370,14 +378,15 @@ export class ChatDragAndDrop extends Themable {
370
378
return variableEntries ;
371
379
}
372
380
373
- private async createImageVariable ( data : Uint8Array , name : string , id ?: string ) {
381
+ private async createImageVariable ( data : Uint8Array , name : string , uri ?: URI , id ?: string , ) : Promise < IChatRequestVariableEntry > {
374
382
return {
375
383
id : id || await imageToHash ( data ) ,
376
384
name : name ,
377
385
value : data ,
378
386
isImage : true ,
379
387
isFile : false ,
380
- isDirectory : false
388
+ isDirectory : false ,
389
+ references : uri ? [ { reference : uri , kind : 'reference' } ] : [ ]
381
390
} ;
382
391
}
383
392
@@ -433,6 +442,45 @@ export class ChatDragAndDrop extends Themable {
433
442
this . overlays . forEach ( overlay => this . updateOverlayStyles ( overlay . overlay ) ) ;
434
443
this . overlayTextBackground = this . getColor ( this . styles . listBackground ) || '' ;
435
444
}
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
+
436
484
}
437
485
438
486
async function getResourceAttachContext ( resource : URI , isDirectory : boolean , textModelService : ITextModelService ) : Promise < IChatRequestVariableEntry | undefined > {
@@ -499,29 +547,3 @@ function symbolId(resource: URI, range?: IRange): string {
499
547
}
500
548
return resource . fsPath + rangePart ;
501
549
}
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