@@ -37,6 +37,7 @@ import { indexOf, isInComment, isMapContainsEmptyPair } from '../utils/astUtils'
37
37
import { isModeline } from './modelineUtil' ;
38
38
import { getSchemaTypeName , isAnyOfAllOfOneOfType , isPrimitiveType } from '../utils/schemaUtils' ;
39
39
import { YamlNode } from '../jsonASTTypes' ;
40
+ import { addIndentationToMultilineString } from '../utils/strings' ;
40
41
41
42
const localize = nls . loadMessageBundle ( ) ;
42
43
@@ -62,6 +63,20 @@ interface CompletionsCollector {
62
63
getNumberOfProposals ( ) : number ;
63
64
result : CompletionList ;
64
65
proposed : { [ key : string ] : CompletionItem } ;
66
+ context : {
67
+ /**
68
+ * The content of the line where the completion is happening.
69
+ */
70
+ lineContent ?: string ;
71
+ /**
72
+ * `true` if the line has a colon.
73
+ */
74
+ hasColon ?: boolean ;
75
+ /**
76
+ * `true` if the line starts with a hyphen.
77
+ */
78
+ hasHyphen ?: boolean ;
79
+ } ;
65
80
}
66
81
67
82
interface InsertText {
@@ -459,6 +474,7 @@ export class YamlCompletion {
459
474
} ,
460
475
result,
461
476
proposed,
477
+ context : { } ,
462
478
} ;
463
479
464
480
if ( this . customTags && this . customTags . length > 0 ) {
@@ -670,6 +686,10 @@ export class YamlCompletion {
670
686
}
671
687
}
672
688
689
+ collector . context . lineContent = lineContent ;
690
+ collector . context . hasColon = lineContent . indexOf ( ':' ) !== - 1 ;
691
+ collector . context . hasHyphen = lineContent . trimStart ( ) . indexOf ( '-' ) === 0 ;
692
+
673
693
// completion for object keys
674
694
if ( node && isMap ( node ) ) {
675
695
// don't suggest properties that are already present
@@ -698,7 +718,7 @@ export class YamlCompletion {
698
718
collector . add ( {
699
719
kind : CompletionItemKind . Property ,
700
720
label : currentWord ,
701
- insertText : this . getInsertTextForProperty ( currentWord , null , '' ) ,
721
+ insertText : this . getInsertTextForProperty ( currentWord , null , '' , collector ) ,
702
722
insertTextFormat : InsertTextFormat . Snippet ,
703
723
} ) ;
704
724
}
@@ -940,6 +960,7 @@ export class YamlCompletion {
940
960
key ,
941
961
propertySchema ,
942
962
separatorAfter ,
963
+ collector ,
943
964
indentCompensation + this . indentation
944
965
) ;
945
966
}
@@ -970,6 +991,7 @@ export class YamlCompletion {
970
991
key ,
971
992
propertySchema ,
972
993
separatorAfter ,
994
+ collector ,
973
995
indentCompensation + this . indentation
974
996
) ,
975
997
insertTextFormat : InsertTextFormat . Snippet ,
@@ -1127,7 +1149,7 @@ export class YamlCompletion {
1127
1149
index ?: number
1128
1150
) : void {
1129
1151
const schemaType = getSchemaTypeName ( schema ) ;
1130
- const insertText = `- ${ this . getInsertTextForObject ( schema , separatorAfter ) . insertText . trimLeft ( ) } ` ;
1152
+ const insertText = `- ${ this . getInsertTextForObject ( schema , separatorAfter , collector ) . insertText . trimLeft ( ) } ` ;
1131
1153
//append insertText to documentation
1132
1154
const schemaTypeTitle = schemaType ? ' type `' + schemaType + '`' : '' ;
1133
1155
const schemaDescription = schema . description ? ' (' + schema . description + ')' : '' ;
@@ -1148,6 +1170,7 @@ export class YamlCompletion {
1148
1170
key : string ,
1149
1171
propertySchema : JSONSchema ,
1150
1172
separatorAfter : string ,
1173
+ collector : CompletionsCollector ,
1151
1174
indent = this . indentation
1152
1175
) : string {
1153
1176
const propertyText = this . getInsertTextForValue ( key , '' , 'string' ) ;
@@ -1218,11 +1241,11 @@ export class YamlCompletion {
1218
1241
nValueProposals += propertySchema . examples . length ;
1219
1242
}
1220
1243
if ( propertySchema . properties ) {
1221
- return `${ resultText } \n${ this . getInsertTextForObject ( propertySchema , separatorAfter , indent ) . insertText } ` ;
1244
+ return `${ resultText } \n${ this . getInsertTextForObject ( propertySchema , separatorAfter , collector , indent ) . insertText } ` ;
1222
1245
} else if ( propertySchema . items ) {
1223
- return ` ${ resultText } \n ${ indent } - ${
1224
- this . getInsertTextForArray ( propertySchema . items , separatorAfter , 1 , indent ) . insertText
1225
- } ` ;
1246
+ let insertText = this . getInsertTextForArray ( propertySchema . items , separatorAfter , collector , 1 , indent ) . insertText ;
1247
+ insertText = resultText + addIndentationToMultilineString ( insertText , `\n ${ indent } - ` , ' ' ) ;
1248
+ return insertText ;
1226
1249
}
1227
1250
if ( nValueProposals === 0 ) {
1228
1251
switch ( type ) {
@@ -1262,10 +1285,30 @@ export class YamlCompletion {
1262
1285
private getInsertTextForObject (
1263
1286
schema : JSONSchema ,
1264
1287
separatorAfter : string ,
1288
+ collector : CompletionsCollector ,
1265
1289
indent = this . indentation ,
1266
1290
insertIndex = 1
1267
1291
) : InsertText {
1268
1292
let insertText = '' ;
1293
+ if ( Array . isArray ( schema . defaultSnippets ) && schema . defaultSnippets . length === 1 ) {
1294
+ const body = schema . defaultSnippets [ 0 ] . body ;
1295
+ if ( isDefined ( body ) ) {
1296
+ let value = this . getInsertTextForSnippetValue (
1297
+ body ,
1298
+ '' ,
1299
+ {
1300
+ newLineFirst : false ,
1301
+ indentFirstObject : false ,
1302
+ shouldIndentWithTab : false ,
1303
+ } ,
1304
+ [ ] ,
1305
+ 0
1306
+ ) ;
1307
+ value = addIndentationToMultilineString ( value , indent , indent ) ;
1308
+
1309
+ return { insertText : value , insertIndex } ;
1310
+ }
1311
+ }
1269
1312
if ( ! schema . properties ) {
1270
1313
insertText = `${ indent } $${ insertIndex ++ } \n` ;
1271
1314
return { insertText, insertIndex } ;
@@ -1306,25 +1349,30 @@ export class YamlCompletion {
1306
1349
}
1307
1350
case 'array' :
1308
1351
{
1309
- const arrayInsertResult = this . getInsertTextForArray ( propertySchema . items , separatorAfter , insertIndex ++ , indent ) ;
1310
- const arrayInsertLines = arrayInsertResult . insertText . split ( '\n' ) ;
1311
- let arrayTemplate = arrayInsertResult . insertText ;
1312
- if ( arrayInsertLines . length > 1 ) {
1313
- for ( let index = 1 ; index < arrayInsertLines . length ; index ++ ) {
1314
- const element = arrayInsertLines [ index ] ;
1315
- arrayInsertLines [ index ] = ` ${ element } ` ;
1316
- }
1317
- arrayTemplate = arrayInsertLines . join ( '\n' ) ;
1318
- }
1352
+ const arrayInsertResult = this . getInsertTextForArray (
1353
+ propertySchema . items ,
1354
+ separatorAfter ,
1355
+ collector ,
1356
+ insertIndex ++ ,
1357
+ indent
1358
+ ) ;
1359
+
1319
1360
insertIndex = arrayInsertResult . insertIndex ;
1320
- insertText += `${ indent } ${ keyEscaped } :\n${ indent } ${ this . indentation } - ${ arrayTemplate } \n` ;
1361
+ insertText +=
1362
+ `${ indent } ${ keyEscaped } :` +
1363
+ addIndentationToMultilineString (
1364
+ arrayInsertResult . insertText ,
1365
+ `\n${ indent } ${ this . indentation } - ` ,
1366
+ `${ this . indentation } `
1367
+ ) ;
1321
1368
}
1322
1369
break ;
1323
1370
case 'object' :
1324
1371
{
1325
1372
const objectInsertResult = this . getInsertTextForObject (
1326
1373
propertySchema ,
1327
1374
separatorAfter ,
1375
+ collector ,
1328
1376
`${ indent } ${ this . indentation } ` ,
1329
1377
insertIndex ++
1330
1378
) ;
@@ -1360,8 +1408,14 @@ export class YamlCompletion {
1360
1408
return { insertText, insertIndex } ;
1361
1409
}
1362
1410
1363
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
1364
- private getInsertTextForArray ( schema : any , separatorAfter : string , insertIndex = 1 , indent = this . indentation ) : InsertText {
1411
+ private getInsertTextForArray (
1412
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
1413
+ schema : any ,
1414
+ separatorAfter : string ,
1415
+ collector : CompletionsCollector ,
1416
+ insertIndex = 1 ,
1417
+ indent = this . indentation
1418
+ ) : InsertText {
1365
1419
let insertText = '' ;
1366
1420
if ( ! schema ) {
1367
1421
insertText = `$${ insertIndex ++ } ` ;
@@ -1389,7 +1443,7 @@ export class YamlCompletion {
1389
1443
break ;
1390
1444
case 'object' :
1391
1445
{
1392
- const objectInsertResult = this . getInsertTextForObject ( schema , separatorAfter , ` ${ indent } ` , insertIndex ++ ) ;
1446
+ const objectInsertResult = this . getInsertTextForObject ( schema , separatorAfter , collector , indent , insertIndex ++ ) ;
1393
1447
insertText = objectInsertResult . insertText . trimLeft ( ) ;
1394
1448
insertIndex = objectInsertResult . insertIndex ;
1395
1449
}
@@ -1591,11 +1645,11 @@ export class YamlCompletion {
1591
1645
indentFirstObject : ! isArray ,
1592
1646
shouldIndentWithTab : ! isArray ,
1593
1647
} ,
1594
- 0 ,
1648
+ arrayDepth ,
1595
1649
isArray
1596
1650
) ;
1597
1651
if ( ! hasProposals && typeof schema . items === 'object' && ! Array . isArray ( schema . items ) ) {
1598
- this . addDefaultValueCompletions ( schema . items , separatorAfter , collector , arrayDepth + 1 ) ;
1652
+ this . addDefaultValueCompletions ( schema . items , separatorAfter , collector , arrayDepth + 1 , true ) ;
1599
1653
}
1600
1654
}
1601
1655
@@ -1656,24 +1710,13 @@ export class YamlCompletion {
1656
1710
if ( Array . isArray ( schema . defaultSnippets ) ) {
1657
1711
for ( const s of schema . defaultSnippets ) {
1658
1712
let type = schema . type ;
1659
- let value = s . body ;
1713
+ const value = s . body ;
1660
1714
let label = s . label ;
1661
1715
let insertText : string ;
1662
1716
let filterText : string ;
1663
1717
if ( isDefined ( value ) ) {
1664
1718
const type = s . type || schema . type ;
1665
- if ( ( arrayDepth === 0 && type === 'array' ) || isArray ) {
1666
- // We know that a - isn't present yet so we need to add one
1667
- const fixedObj = { } ;
1668
- Object . keys ( value ) . forEach ( ( val , index ) => {
1669
- if ( index === 0 && ! val . startsWith ( '-' ) ) {
1670
- fixedObj [ `- ${ val } ` ] = value [ val ] ;
1671
- } else {
1672
- fixedObj [ ` ${ val } ` ] = value [ val ] ;
1673
- }
1674
- } ) ;
1675
- value = fixedObj ;
1676
- }
1719
+
1677
1720
const existingProps = Object . keys ( collector . proposed ) . filter (
1678
1721
( proposedProp ) => collector . proposed [ proposedProp ] . label === existingProposeItem
1679
1722
) ;
@@ -1683,6 +1726,24 @@ export class YamlCompletion {
1683
1726
if ( insertText === '' && value ) {
1684
1727
continue ;
1685
1728
}
1729
+
1730
+ if ( ( arrayDepth === 0 && type === 'array' ) || isArray ) {
1731
+ // add extra hyphen if we are in array, but the hyphen is missing on current line
1732
+ // but don't add it for array value because it's already there from getInsertTextForSnippetValue
1733
+ const addHyphen = ! collector . context . hasHyphen && ! Array . isArray ( value ) ? '- ' : '' ;
1734
+ // add new line if the cursor is after the colon
1735
+ const addNewLine = collector . context . hasColon ? `\n${ this . indentation } ` : '' ;
1736
+ // add extra indent if new line and hyphen are added
1737
+ const addIndent = isArray && addNewLine && addHyphen ? this . indentation : '' ;
1738
+ // const addIndent = addHyphen && addNewLine ? this.indentation : '';
1739
+
1740
+ insertText = addIndentationToMultilineString (
1741
+ insertText . trimStart ( ) ,
1742
+ `${ addNewLine } ${ addHyphen } ` ,
1743
+ `${ addIndent } ${ this . indentation } `
1744
+ ) ;
1745
+ }
1746
+
1686
1747
label = label || this . getLabelForSnippetValue ( value ) ;
1687
1748
} else if ( typeof s . bodyText === 'string' ) {
1688
1749
let prefix = '' ,
0 commit comments