patch 9.1.1590: cannot perform autocompletion
Problem:  cannot perform autocompletion
Solution: Add the 'autocomplete' option value
          (Girish Palya)
This change introduces the 'autocomplete' ('ac') boolean option to
enable automatic popup menu completion during insert mode. When enabled,
Vim shows a completion menu as you type, similar to pressing |i\_CTRL-N|
manually. The items are collected from sources defined in the
'complete' option.
To ensure responsiveness, this feature uses a time-sliced strategy:
- Sources earlier in the 'complete' list are given more time.
- If a source exceeds its allocated timeout, it is interrupted.
- The next source is then started with a reduced timeout (exponentially
  decayed).
- A small minimum ensures every source still gets a brief chance to
  contribute.
The feature is fully compatible with other |i_CTRL-X| completion modes,
which can temporarily suspend automatic completion when triggered.
See :help 'autocomplete' and :help ins-autocompletion for more details.
To try it out, use :set ac
You should see a popup menu appear automatically with suggestions. This
works seamlessly across:
- Large files (multi-gigabyte size)
- Massive codebases (:argadd thousands of .c or .h files)
- Large dictionaries via the `k` option
- Slow or blocking LSP servers or user-defined 'completefunc'
Despite potential slowness in sources, the menu remains fast,
responsive, and useful.
Compatibility: This mode is fully compatible with existing completion
methods. You can still invoke any CTRL-X based completion (e.g.,
CTRL-X CTRL-F for filenames) at any time (CTRL-X temporarily
suspends 'autocomplete'). To specifically use i_CTRL-N, dismiss the
current popup by pressing CTRL-E first.
---
How it works
To keep completion snappy under all conditions, autocompletion uses a
decaying time-sliced algorithm:
- Starts with an initial timeout (80ms).
- If a source does not complete within the timeout, it's interrupted and
  the timeout is halved for the next source.
- This continues recursively until a minimum timeout (5ms) is reached.
- All sources are given a chance, but slower ones are de-prioritized
  quickly.
Most of the time, matches are computed well within the initial window.
---
Implementation details
- Completion logic is mostly triggered in `edit.c` and handled in
  insexpand.c.
- Uses existing inc_compl_check_keys() mechanism, so no new polling
  hooks are needed.
- The completion system already checks for user input periodically; it
  now also checks for timer expiry.
---
Design notes
- The menu doesn't continuously update after it's shown to prevent
  visual distraction (due to resizing) and ensure the internal list
  stays synchronized with the displayed menu.
- The 'complete' option determines priority—sources listed earlier get
  more time.
- The exponential time-decay mechanism prevents indefinite collection,
  contributing to low CPU usage and a minimal memory footprint.
- Timeout values are intentionally not configurable—this system is
  optimized to "just work" out of the box. If autocompletion feels slow,
  it typically indicates a deeper performance bottleneck (e.g., a slow
  custom function not using `complete_check()`) rather than a
  configuration issue.
---
Performance
Based on testing, the total roundtrip time for completion is generally
under 200ms. For common usage, it often responds in under 50ms on an
average laptop, which falls within the "feels instantaneous" category
(sub-100ms) for perceived user experience.
| Upper Bound (ms) | Perceived UX
|----------------- |-------------
| <100 ms          | Excellent; instantaneous
| <200 ms          | Good; snappy
| >300 ms          | Noticeable lag
| >500 ms          | Sluggish/Broken
---
Why this belongs in core:
- Minimal and focused implementation, tightly integrated with existing
  Insert-mode completion logic.
- Zero reliance on autocommands and external scripting.
- Makes full use of Vim’s highly composable 'complete' infrastructure
  while avoiding the complexity of plugin-based solutions.
- Gives users C native autocompletion with excellent responsiveness and
  no configuration overhead.
- Adds a key UX functionality in a simple, performant, and Vim-like way.
closes: #17812
Signed-off-by: Girish Palya <girishji@gmail.com>
Signed-off-by: Christian Brabandt <cb@256bit.org>
			
			
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							44309b9d08
						
					
				
				
					commit
					af9a7a04f1
				
			
							
								
								
									
										23
									
								
								src/edit.c
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								src/edit.c
									
									
									
									
									
								
							| @ -983,6 +983,17 @@ doESCkey: | ||||
| 	case Ctrl_H: | ||||
| 	    did_backspace = ins_bs(c, BACKSPACE_CHAR, &inserted_space); | ||||
| 	    auto_format(FALSE, TRUE); | ||||
| 	    if (did_backspace && p_ac && !char_avail() | ||||
| 		    && curwin->w_cursor.col > 0) | ||||
| 	    { | ||||
| 		c = char_before_cursor(); | ||||
| 		if (ins_compl_setup_autocompl(c)) | ||||
| 		{ | ||||
| 		    update_screen(UPD_VALID); // Show char deletion immediately | ||||
| 		    out_flush(); | ||||
| 		    goto docomplete; // Trigger autocompletion | ||||
| 		} | ||||
| 	    } | ||||
| 	    break; | ||||
|  | ||||
| 	case Ctrl_W:	// delete word before the cursor | ||||
| @ -1401,6 +1412,14 @@ normalchar: | ||||
| 	    // closed fold. | ||||
| 	    foldOpenCursor(); | ||||
| #endif | ||||
| 	    // Trigger autocompletion | ||||
| 	    if (p_ac && !char_avail() && ins_compl_setup_autocompl(c)) | ||||
| 	    { | ||||
| 		update_screen(UPD_VALID); // Show character immediately | ||||
| 		out_flush(); | ||||
| 		goto docomplete; | ||||
| 	    } | ||||
|  | ||||
| 	    break; | ||||
| 	}   // end of switch (c) | ||||
|  | ||||
| @ -2233,6 +2252,10 @@ insertchar( | ||||
|     if (       !ISSPECIAL(c) | ||||
| 	    && (!has_mbyte || (*mb_char2len)(c) == 1) | ||||
| 	    && !has_insertcharpre() | ||||
| #ifdef FEAT_EVAL | ||||
| 	    // Skip typeahead if test_override("char_avail", 1) was called. | ||||
| 	    && !disable_char_avail_for_testing | ||||
| #endif | ||||
| 	    && vpeekc() != NUL | ||||
| 	    && !(State & REPLACE_FLAG) | ||||
| 	    && !cindent_on() | ||||
|  | ||||
		Reference in New Issue
	
	Block a user