@@ -12,13 +12,15 @@ import '../../flutter_code_editor.dart';
12
12
import '../autocomplete/autocompleter.dart' ;
13
13
import '../code/code_edit_result.dart' ;
14
14
import '../code/key_event.dart' ;
15
+ import '../code_modifiers/insertion.dart' ;
15
16
import '../history/code_history_controller.dart' ;
16
17
import '../history/code_history_record.dart' ;
17
18
import '../search/controller.dart' ;
18
19
import '../search/result.dart' ;
19
20
import '../search/search_navigation_controller.dart' ;
20
21
import '../search/settings_controller.dart' ;
21
22
import '../single_line_comments/parser/single_line_comments.dart' ;
23
+ import '../util/string_util.dart' ;
22
24
import '../wip/autocomplete/popup_controller.dart' ;
23
25
import 'actions/comment_uncomment.dart' ;
24
26
import 'actions/copy.dart' ;
@@ -28,6 +30,7 @@ import 'actions/indent.dart';
28
30
import 'actions/outdent.dart' ;
29
31
import 'actions/redo.dart' ;
30
32
import 'actions/search.dart' ;
33
+ import 'actions/tab.dart' ;
31
34
import 'actions/undo.dart' ;
32
35
import 'search_result_highlighted_builder.dart' ;
33
36
import 'span_builder.dart' ;
@@ -50,6 +53,7 @@ class CodeController extends TextEditingController {
50
53
/// Calls [AbstractAnalyzer.analyze] after change with 500ms debounce.
51
54
AbstractAnalyzer get analyzer => _analyzer;
52
55
AbstractAnalyzer _analyzer;
56
+
53
57
set analyzer (AbstractAnalyzer analyzer) {
54
58
if (_analyzer == analyzer) {
55
59
return ;
@@ -107,6 +111,7 @@ class CodeController extends TextEditingController {
107
111
108
112
SearchSettingsController get _searchSettingsController =>
109
113
searchController.settingsController;
114
+
110
115
SearchNavigationController get _searchNavigationController =>
111
116
searchController.navigationController;
112
117
@@ -130,8 +135,21 @@ class CodeController extends TextEditingController {
130
135
SearchIntent : SearchAction (controller: this ),
131
136
DismissIntent : CustomDismissAction (controller: this ),
132
137
EnterKeyIntent : EnterKeyAction (controller: this ),
138
+ TabKeyIntent : TabKeyAction (controller: this ),
133
139
};
134
140
141
+ static const defaultCodeModifiers = [
142
+ IndentModifier (),
143
+ CloseBlockModifier (),
144
+ TabModifier (),
145
+ InsertionCodeModifier .backticks,
146
+ InsertionCodeModifier .braces,
147
+ InsertionCodeModifier .brackets,
148
+ InsertionCodeModifier .doubleQuotes,
149
+ InsertionCodeModifier .parentheses,
150
+ InsertionCodeModifier .singleQuotes,
151
+ ];
152
+
135
153
CodeController ({
136
154
String ? text,
137
155
Mode ? language,
@@ -143,11 +161,7 @@ class CodeController extends TextEditingController {
143
161
this .patternMap,
144
162
this .readOnly = false ,
145
163
this .params = const EditorParams (),
146
- this .modifiers = const [
147
- IndentModifier (),
148
- CloseBlockModifier (),
149
- TabModifier (),
150
- ],
164
+ this .modifiers = defaultCodeModifiers,
151
165
}) : _analyzer = analyzer,
152
166
_readOnlySectionNames = readOnlySectionNames,
153
167
_code = Code .empty,
@@ -347,30 +361,60 @@ class CodeController extends TextEditingController {
347
361
insertStr ('\n ' );
348
362
}
349
363
364
+ void onTabKeyAction () {
365
+ if (popupController.shouldShow) {
366
+ insertSelectedWord ();
367
+ return ;
368
+ }
369
+
370
+ insertStr (' ' * params.tabSpaces);
371
+ }
372
+
350
373
/// Inserts the word selected from the list of completions
351
374
void insertSelectedWord () {
352
375
final previousSelection = selection;
353
376
final selectedWord = popupController.getSelectedWord ();
354
377
final startPosition = value.wordAtCursorStart;
378
+ final currentWord = value.wordAtCursor;
355
379
356
- if (startPosition != null ) {
357
- final replacedText = text.replaceRange (
358
- startPosition,
359
- selection.baseOffset,
360
- selectedWord,
361
- );
380
+ if (startPosition == null || currentWord == null ) {
381
+ popupController.hide ();
382
+ return ;
383
+ }
362
384
363
- final adjustedSelection = previousSelection.copyWith (
364
- baseOffset: startPosition + selectedWord.length,
365
- extentOffset: startPosition + selectedWord.length,
366
- );
385
+ final endReplacingPosition = startPosition + currentWord.length;
386
+ final endSelectionPosition = startPosition + selectedWord.length;
367
387
368
- value = TextEditingValue (
369
- text: replacedText,
370
- selection: adjustedSelection,
371
- );
388
+ var additionalSpaceIfEnd = '' ;
389
+ var offsetIfEndsWithSpace = 1 ;
390
+ if (text.length < endReplacingPosition + 1 ) {
391
+ additionalSpaceIfEnd = ' ' ;
392
+ } else {
393
+ final charAfterText = text[endReplacingPosition];
394
+ if (charAfterText != ' ' &&
395
+ ! StringUtil .isDigit (charAfterText) &&
396
+ ! StringUtil .isLetterEng (charAfterText)) {
397
+ // ex. case ';' or other finalizer, or symbol
398
+ offsetIfEndsWithSpace = 0 ;
399
+ }
372
400
}
373
401
402
+ final replacedText = text.replaceRange (
403
+ startPosition,
404
+ endReplacingPosition,
405
+ '$selectedWord $additionalSpaceIfEnd ' ,
406
+ );
407
+
408
+ final adjustedSelection = previousSelection.copyWith (
409
+ baseOffset: endSelectionPosition + offsetIfEndsWithSpace,
410
+ extentOffset: endSelectionPosition + offsetIfEndsWithSpace,
411
+ );
412
+
413
+ value = TextEditingValue (
414
+ text: replacedText,
415
+ selection: adjustedSelection,
416
+ );
417
+
374
418
popupController.hide ();
375
419
}
376
420
0 commit comments