diff --git a/.circleci/config.yml b/.circleci/config.yml
deleted file mode 100644
index 8de098f..0000000
--- a/.circleci/config.yml
+++ /dev/null
@@ -1,16 +0,0 @@
-version: 2
-jobs:
- build:
- parallelism: 4
- shell: /bin/bash --login
- docker:
- - image: ericfreese/zsh-autosuggestions-test:latest
- command: /sbin/init
- steps:
- - checkout
- - run:
- name: Running tests
- command: |
- for v in $(grep "^[^#]" ZSH_VERSIONS | awk "(NR + $CIRCLE_NODE_INDEX) % $CIRCLE_NODE_TOTAL == 0"); do
- TEST_ZSH_BIN=zsh-$v make test || exit 1
- done
diff --git a/.github/ISSUE_TEMPLATE/bug-report.md b/.github/ISSUE_TEMPLATE/bug-report.md
new file mode 100644
index 0000000..7663df6
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/bug-report.md
@@ -0,0 +1,36 @@
+---
+name: Bug report
+about: Create a report to help us improve
+title: ''
+labels: bug
+assignees: ''
+
+---
+
+### Describe the bug
+
+
+### To Reproduce
+Steps to reproduce the behavior:
+
+
+
+```sh
+% zsh -df
+% source path/to/zsh-autosuggestions.zsh
+% ... # what do you do to reproduce?
+```
+
+### Expected behavior
+
+
+### Screenshots
+
+
+### Desktop
+ - OS + distribution:
+ - Zsh version:
+ - Plugin version:
+
+### Additional context
+
diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md
new file mode 100644
index 0000000..5874625
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/feature_request.md
@@ -0,0 +1,20 @@
+---
+name: Feature request
+about: Suggest an idea for this project
+title: ''
+labels: enhancement
+assignees: ''
+
+---
+
+### Is your feature request related to a problem? Please describe.
+
+
+### Describe the solution you'd like
+
+
+### Describe alternatives you've considered
+
+
+### Additional context
+
diff --git a/.github/workflows/integration.yml b/.github/workflows/integration.yml
new file mode 100644
index 0000000..ec7bc75
--- /dev/null
+++ b/.github/workflows/integration.yml
@@ -0,0 +1,51 @@
+on: [push, pull_request]
+concurrency:
+ group: ${{ github.workflow }}-${{ github.ref }}
+ cancel-in-progress: true
+env:
+ IMAGE_CACHE_PATH: /tmp/.image-cache
+ IMAGE_CACHE_NAME: zsh-autosuggestions-test
+jobs:
+ determine-versions:
+ runs-on: ubuntu-22.04
+ outputs:
+ versions: ${{ steps.set-versions.outputs.versions }}
+ steps:
+ - uses: actions/checkout@v3
+ - id: set-versions
+ run: |
+ echo "versions=$(
+ grep "^[^#]" ZSH_VERSIONS \
+ | sed -E 's/(^|$)/"/g' \
+ | paste -sd ',' - \
+ | sed -e 's/^/[/' -e 's/$/]/'
+ )" >> $GITHUB_OUTPUT
+ test:
+ needs: determine-versions
+ runs-on: ubuntu-22.04
+ strategy:
+ matrix:
+ version: ${{ fromJson(needs.determine-versions.outputs.versions) }}
+ steps:
+ - uses: actions/checkout@v3
+ - name: Docker image cache
+ id: image-cache
+ uses: actions/cache@v3
+ with:
+ path: ${{ env.IMAGE_CACHE_PATH }}
+ key: image-cache-${{ matrix.version }}-${{ hashFiles('Dockerfile', 'install_test_zsh.sh', 'Gemfile.lock') }}
+ - name: Load cached docker image if available
+ if: ${{ steps.image-cache.outputs.cache-hit }}
+ run: gunzip < $IMAGE_CACHE_PATH/$IMAGE_CACHE_NAME.tar.gz | docker load
+ - name: Build the docker image if necessary
+ if: ${{ !steps.image-cache.outputs.cache-hit }}
+ run: |
+ docker build --build-arg TEST_ZSH_VERSION=${{ matrix.version }} -t $IMAGE_CACHE_NAME .
+ mkdir -p $IMAGE_CACHE_PATH
+ docker save $IMAGE_CACHE_NAME | gzip > $IMAGE_CACHE_PATH/$IMAGE_CACHE_NAME.tar.gz
+ - name: Run the tests
+ run: |
+ docker run --rm \
+ -v $PWD:/zsh-autosuggestions \
+ $IMAGE_CACHE_NAME \
+ make test
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..d8decde
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,2 @@
+# zsh word code files
+*.zwc
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 37bf991..30c7735 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,23 @@
# Changelog
+## v0.7.1
+- Clear POSTDISPLAY instead of unsetting (#634)
+- Always reset async file descriptor after consuming it (#630)
+- Always use builtin `exec` (#628)
+- Add `history-beginning-search-*-end` widgets to clear widget list (#619)
+- Switch CI from Circle CI to GitHub Actions
+
+## v0.7.0
+- Enable asynchronous mode by default (#498)
+- No longer wrap user widgets starting with `autosuggest-` prefix (#496)
+- Fix a bug wrapping widgets that modify the buffer (#541)
+
+
+## v0.6.4
+- Fix `vi-forward-char` triggering a bell when using it to accept a suggestion (#488)
+- New configuration option to skip completion suggestions when buffer matches a pattern (#487)
+- New configuration option to ignore history entries matching a pattern (#456)
+
## v0.6.3
- Fixed bug moving cursor to end of buffer after accepting suggestion (#453)
diff --git a/Dockerfile b/Dockerfile
index 0d51407..f5dd3c4 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -1,5 +1,8 @@
FROM ruby:2.5.3-alpine
+ARG TEST_ZSH_VERSION
+RUN : "${TEST_ZSH_VERSION:?}"
+
RUN apk add --no-cache autoconf
RUN apk add --no-cache libtool
RUN apk add --no-cache libcap-dev
@@ -11,10 +14,8 @@ RUN apk add --no-cache tmux
WORKDIR /zsh-autosuggestions
-ADD ZSH_VERSIONS /zsh-autosuggestions/ZSH_VERSIONS
-ADD install_test_zsh.sh /zsh-autosuggestions/install_test_zsh.sh
+ADD install_test_zsh.sh ./
RUN ./install_test_zsh.sh
-ADD Gemfile /zsh-autosuggestions/Gemfile
-ADD Gemfile.lock /zsh-autosuggestions/Gemfile.lock
+ADD Gemfile Gemfile.lock ./
RUN bundle install
diff --git a/INSTALL.md b/INSTALL.md
index d06db2b..7f0a395 100644
--- a/INSTALL.md
+++ b/INSTALL.md
@@ -1,26 +1,25 @@
# Installation
-* [Manual](#manual-git-clone)
+* [Packages](#packages)
* [Antigen](#antigen)
* [Oh My Zsh](#oh-my-zsh)
-* [Arch Linux](#arch-linux)
-* [macOS via Homebrew](#macos-via-homebrew)
+* [HomeBrew](#homebrew)
+* [Manual](#manual-git-clone)
-## Manual (Git Clone)
+## Packages
-1. Clone this repository somewhere on your machine. This guide will assume `~/.zsh/zsh-autosuggestions`.
-
- ```sh
- git clone https://github.com/zsh-users/zsh-autosuggestions ~/.zsh/zsh-autosuggestions
- ```
-
-2. Add the following to your `.zshrc`:
-
- ```sh
- source ~/.zsh/zsh-autosuggestions/zsh-autosuggestions.zsh
- ```
-
-3. Start a new terminal session.
+| System | Package |
+| ------------- | ------------- |
+| Alpine Linux | [zsh-autosuggestions](https://pkgs.alpinelinux.org/packages?name=zsh-autosuggestions) |
+| Debian / Ubuntu | [zsh-autosuggestions OBS repository](https://software.opensuse.org/download.html?project=shells%3Azsh-users%3Azsh-autosuggestions&package=zsh-autosuggestions) |
+| Fedora / CentOS / RHEL / Scientific Linux | [zsh-autosuggestions OBS repository](https://software.opensuse.org/download.html?project=shells%3Azsh-users%3Azsh-autosuggestions&package=zsh-autosuggestions) |
+| OpenSUSE / SLE | [zsh-autosuggestions OBS repository](https://software.opensuse.org/download.html?project=shells%3Azsh-users%3Azsh-autosuggestions&package=zsh-autosuggestions) |
+| Arch Linux / Manjaro / Antergos / Hyperbola | [zsh-autosuggestions](https://www.archlinux.org/packages/zsh-autosuggestions), [zsh-autosuggestions-git](https://aur.archlinux.org/packages/zsh-autosuggestions-git) |
+| NixOS | [zsh-autosuggestions](https://github.com/NixOS/nixpkgs/blob/master/pkgs/by-name/zs/zsh-autosuggestions/package.nix) |
+| Void Linux | [zsh-autosuggestions](https://github.com/void-linux/void-packages/blob/master/srcpkgs/zsh-autosuggestions/template) |
+| Mac OS | [homebrew](https://formulae.brew.sh/formula/zsh-autosuggestions) |
+| NetBSD | [pkgsrc](http://ftp.netbsd.org/pub/pkgsrc/current/pkgsrc/shells/zsh-autosuggestions/README.html) |
+| FreeBSD | [pkg](https://cgit.freebsd.org/ports/tree/shells/zsh-autosuggestions) |
## Antigen
@@ -43,41 +42,41 @@
2. Add the plugin to the list of plugins for Oh My Zsh to load (inside `~/.zshrc`):
```sh
- plugins=(zsh-autosuggestions)
+ plugins=(
+ # other plugins...
+ zsh-autosuggestions
+ )
```
3. Start a new terminal session.
-## Arch Linux
-
-1. Install [`zsh-autosuggestions`](https://www.archlinux.org/packages/community/any/zsh-autosuggestions/) from the `community` repository.
-
- ```sh
- pacman -S zsh-autosuggestions
- ```
-
- or, to use a package based on the `master` branch, install [`zsh-autosuggestions-git`](https://aur.archlinux.org/packages/zsh-autosuggestions-git/) from the [AUR](https://wiki.archlinux.org/index.php/Arch_User_Repository).
-
-2. Add the following to your `.zshrc`:
-
- ```sh
- source /usr/share/zsh/plugins/zsh-autosuggestions/zsh-autosuggestions.zsh
- ```
-
-3. Start a new terminal session.
-
-## macOS via Homebrew
-
-1. Install the `zsh-autosuggestions` package using [Homebrew](https://brew.sh/).
+## Homebrew
+1. Install command:
```sh
brew install zsh-autosuggestions
```
-2. Add the following to your `.zshrc`:
-
+2. To activate the autosuggestions, add the following at the end of your .zshrc:
+
```sh
- source /usr/local/share/zsh-autosuggestions/zsh-autosuggestions.zsh
+ source $(brew --prefix)/share/zsh-autosuggestions/zsh-autosuggestions.zsh
+ ```
+
+3. Start a new terminal session.
+
+## Manual (Git Clone)
+
+1. Clone this repository somewhere on your machine. This guide will assume `~/.zsh/zsh-autosuggestions`.
+
+ ```sh
+ git clone https://github.com/zsh-users/zsh-autosuggestions ~/.zsh/zsh-autosuggestions
+ ```
+
+2. Add the following to your `.zshrc`:
+
+ ```sh
+ source ~/.zsh/zsh-autosuggestions/zsh-autosuggestions.zsh
```
3. Start a new terminal session.
diff --git a/LICENSE b/LICENSE
index ef7cfb6..7ea78cc 100644
--- a/LICENSE
+++ b/LICENSE
@@ -1,5 +1,5 @@
Copyright (c) 2013 Thiago de Arruda
-Copyright (c) 2016-2019 Eric Freese
+Copyright (c) 2016-2021 Eric Freese
Permission is hereby granted, free of charge, to any person
obtaining a copy of this software and associated documentation
diff --git a/Makefile b/Makefile
index f6d13a7..6f5431e 100644
--- a/Makefile
+++ b/Makefile
@@ -1,4 +1,4 @@
-SRC_DIR := ./src
+SRC_DIR := ./src
SRC_FILES := \
$(SRC_DIR)/config.zsh \
diff --git a/README.md b/README.md
index fad9bf5..a8c1b6c 100644
--- a/README.md
+++ b/README.md
@@ -6,7 +6,7 @@ It suggests commands as you type based on history and completions.
Requirements: Zsh v4.3.11 or later
-[](https://circleci.com/gh/zsh-users/zsh-autosuggestions)
+[](https://gitter.im/zsh-users/zsh-autosuggestions)
@@ -34,7 +34,7 @@ You may want to override the default global config variables. Default values of
### Suggestion Highlight Style
-Set `ZSH_AUTOSUGGEST_HIGHLIGHT_STYLE` to configure the style that the suggestion is shown with. The default is `fg=8`, which will set the foreground color to color 8 from the 256-color palette. If your terminal only supports 8 colors, you will need to use a number between 0 and 7.
+Set `ZSH_AUTOSUGGEST_HIGHLIGHT_STYLE` to configure the style that the suggestion is shown with. The default is `fg=8`, which will set the foreground color to color 8 from the [256-color palette](https://upload.wikimedia.org/wikipedia/commons/1/15/Xterm_256color_chart.svg). If your terminal only supports 8 colors, you will need to use a number between 0 and 7.
Background color can also be set, and the suggestion can be styled bold, underlined, or standout. For example, this would show suggestions with bold, underlined, pink text on a cyan background:
@@ -52,8 +52,8 @@ For more info, read the Character Highlighting section of the zsh manual: `man z
`ZSH_AUTOSUGGEST_STRATEGY` is an array that specifies how suggestions should be generated. The strategies in the array are tried successively until a suggestion is found. There are currently three built-in strategies to choose from:
- `history`: Chooses the most recent match from history.
+- `completion`: Chooses a suggestion based on what tab-completion would suggest. (requires `zpty` module, which is included with zsh since 4.0.1)
- `match_prev_cmd`: Like `history`, but chooses the most recent match whose preceding history item matches the most recently executed command ([more info](src/strategies/match_prev_cmd.zsh)). Note that this strategy won't work as expected with ZSH options that don't preserve the history order such as `HIST_IGNORE_ALL_DUPS` or `HIST_EXPIRE_DUPS_FIRST`.
-- `completion`: (experimental) Chooses a suggestion based on what tab-completion would suggest. (requires `zpty` module)
For example, setting `ZSH_AUTOSUGGEST_STRATEGY=(history completion)` will first try to find a suggestion from your history, but, if it can't find a match, will find a suggestion from the completion engine.
@@ -78,14 +78,28 @@ Widgets that modify the buffer and are not found in any of these arrays will fet
Set `ZSH_AUTOSUGGEST_BUFFER_MAX_SIZE` to an integer value to disable autosuggestion for large buffers. The default is unset, which means that autosuggestion will be tried for any buffer size. Recommended value is 20.
This can be useful when pasting large amount of text in the terminal, to avoid triggering autosuggestion for strings that are too long.
-### Enable Asynchronous Mode
+### Asynchronous Mode
-As of `v0.4.0`, suggestions can be fetched asynchronously. To enable this behavior, set the `ZSH_AUTOSUGGEST_USE_ASYNC` variable (it can be set to anything).
+Suggestions are fetched asynchronously by default in zsh versions 5.0.8 and greater. To disable asynchronous suggestions and fetch them synchronously instead, `unset ZSH_AUTOSUGGEST_USE_ASYNC` after sourcing the plugin.
+
+Alternatively, if you are using a version of zsh older than 5.0.8 and want to enable asynchronous mode, set the `ZSH_AUTOSUGGEST_USE_ASYNC` variable after sourcing the plugin (it can be set to anything). Note that there is [a bug](https://github.com/zsh-users/zsh-autosuggestions/issues/364#issuecomment-481423232) in versions of zsh older than 5.0.8 where ctrl + c will fail to reset the prompt immediately after fetching a suggestion asynchronously.
### Disabling automatic widget re-binding
Set `ZSH_AUTOSUGGEST_MANUAL_REBIND` (it can be set to anything) to disable automatic widget re-binding on each precmd. This can be a big boost to performance, but you'll need to handle re-binding yourself if any of the widget lists change or if you or another plugin wrap any of the autosuggest widgets. To re-bind widgets, run `_zsh_autosuggest_bind_widgets`.
+### Ignoring history suggestions that match a pattern
+
+Set `ZSH_AUTOSUGGEST_HISTORY_IGNORE` to a [glob pattern](http://zsh.sourceforge.net/Doc/Release/Expansion.html#Glob-Operators) to prevent offering suggestions for history entries that match the pattern. For example, set it to `"cd *"` to never suggest any `cd` commands from history. Or set to `"?(#c50,)"` to never suggest anything 50 characters or longer.
+
+**Note:** This only affects the `history` and `match_prev_cmd` suggestion strategies.
+
+### Skipping completion suggestions for certain cases
+
+Set `ZSH_AUTOSUGGEST_COMPLETION_IGNORE` to a [glob pattern](http://zsh.sourceforge.net/Doc/Release/Expansion.html#Glob-Operators) to prevent offering completion suggestions when the buffer matches that pattern. For example, set it to `"git *"` to disable completion suggestions for git subcommands.
+
+**Note:** This only affects the `completion` suggestion strategy.
+
### Key Bindings
@@ -155,18 +169,16 @@ Tests are written in ruby using the [`rspec`](http://rspec.info/) framework. The
Test files live in `spec/`. To run the tests, run `make test`. To run a specific test, run `TESTS=spec/some_spec.rb make test`. You can also specify a `zsh` binary to use by setting the `TEST_ZSH_BIN` environment variable (ex: `TEST_ZSH_BIN=/bin/zsh make test`).
-A docker image for testing is available [on docker hub](https://hub.docker.com/r/ericfreese/zsh-autosuggestions-test). It comes with ruby, the bundler dependencies, and all supported versions of zsh installed.
-
-Pull the docker image with:
+It's possible to run the tests for any supported version of zsh in a Docker image by building an image from the provided Dockerfile. To build the docker image for a specific version of zsh (where `` below is substituted with the contents of a line from the [`ZSH_VERSIONS`](ZSH_VERSIONS) file), run:
```sh
-docker pull ericfreese/zsh-autosuggestions-test
+docker build --build-arg TEST_ZSH_VERSION= -t zsh-autosuggestions-test .
```
-To run the tests for a specific version of zsh (where `` below is substituted with the contents of a line from the [`ZSH_VERSIONS`](ZSH_VERSIONS) file):
+After building the image, run the tests via:
```sh
-docker run -it -e TEST_ZSH_BIN=zsh- -v $PWD:/zsh-autosuggestions zsh-autosuggestions-test make test
+docker run -it -v $PWD:/zsh-autosuggestions zsh-autosuggestions-test make test
```
diff --git a/VERSION b/VERSION
index e4c57af..63f2359 100644
--- a/VERSION
+++ b/VERSION
@@ -1 +1 @@
-v0.6.3
+v0.7.1
diff --git a/ZSH_VERSIONS b/ZSH_VERSIONS
index ed7b882..23006db 100644
--- a/ZSH_VERSIONS
+++ b/ZSH_VERSIONS
@@ -1,9 +1,5 @@
# Zsh releases to run tests against
# See https://github.com/zsh-users/zsh/releases
-#
-# When modifying this file, rebuild and push docker image:
-# $ docker build -t ericfreese/zsh-autosuggestions-test .
-# $ docker push ericfreese/zsh-autosuggestions-test
4.3.11
5.0.2
5.0.8
@@ -14,3 +10,5 @@
5.5.1
5.6.2
5.7.1
+5.8.1
+5.9
diff --git a/install_test_zsh.sh b/install_test_zsh.sh
index 40dc4c5..6cac9f5 100755
--- a/install_test_zsh.sh
+++ b/install_test_zsh.sh
@@ -2,25 +2,22 @@
set -ex
-for v in $(grep "^[^#]" ZSH_VERSIONS); do
- mkdir zsh-$v
- cd zsh-$v
+mkdir zsh-build
+cd zsh-build
- curl -L https://api.github.com/repos/zsh-users/zsh/tarball/zsh-$v | tar xz --strip=1
+curl -L https://api.github.com/repos/zsh-users/zsh/tarball/zsh-$TEST_ZSH_VERSION | tar xz --strip=1
- ./Util/preconfig
- ./configure --enable-pcre \
- --enable-cap \
- --enable-multibyte \
- --with-term-lib='ncursesw tinfo' \
- --with-tcsetpgrp \
- --program-suffix="-$v"
+./Util/preconfig
+./configure --enable-pcre \
+ --enable-cap \
+ --enable-multibyte \
+ --with-term-lib='ncursesw tinfo' \
+ --with-tcsetpgrp
- make install.bin
- make install.modules
- make install.fns
+make install.bin
+make install.modules
+make install.fns
- cd ..
+cd ..
- rm -rf zsh-$v
-done
+rm -rf zsh-build
diff --git a/spec/multi_line_spec.rb b/spec/multi_line_spec.rb
index 4ff2ae1..364780a 100644
--- a/spec/multi_line_spec.rb
+++ b/spec/multi_line_spec.rb
@@ -1,11 +1,6 @@
describe 'a multi-line suggestion' do
it 'should be displayed on multiple lines' do
- with_history(-> {
- session.send_string('echo "')
- session.send_keys('enter')
- session.send_string('"')
- session.send_keys('enter')
- }) do
+ with_history("echo \"\n\"") do
session.send_keys('e')
wait_for { session.content }.to eq("echo \"\n\"")
end
diff --git a/spec/options/use_async_spec.rb b/spec/options/use_async_spec.rb
deleted file mode 100644
index 420dcc3..0000000
--- a/spec/options/use_async_spec.rb
+++ /dev/null
@@ -1,7 +0,0 @@
-describe 'suggestion fetching' do
- it 'is performed synchronously'
-
- context 'when ZSH_AUTOSUGGEST_USE_ASYNC is set' do
- it 'is performed asynchronously'
- end
-end
diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb
index cb149ef..dc1abb0 100644
--- a/spec/spec_helper.rb
+++ b/spec/spec_helper.rb
@@ -1,6 +1,7 @@
require 'pry'
require 'rspec/wait'
require 'terminal_session'
+require 'tempfile'
RSpec.shared_context 'terminal session' do
let(:term_opts) { {} }
@@ -21,18 +22,20 @@ RSpec.shared_context 'terminal session' do
end
def with_history(*commands, &block)
- session.run_command('fc -p')
+ Tempfile.create do |f|
+ f.write(commands.map{|c| c.gsub("\n", "\\\n")}.join("\n"))
+ f.flush
- commands.each do |c|
- c.respond_to?(:call) ? c.call : session.run_command(c)
+ session.run_command('fc -p')
+ session.run_command("fc -R #{f.path}")
+
+ session.clear_screen
+
+ yield block
+
+ session.send_keys('C-c')
+ session.run_command('fc -P')
end
-
- session.clear_screen
-
- yield block
-
- session.send_keys('C-c')
- session.run_command('fc -P')
end
end
diff --git a/spec/strategies/completion_spec.rb b/spec/strategies/completion_spec.rb
index 2be358a..92794d6 100644
--- a/spec/strategies/completion_spec.rb
+++ b/spec/strategies/completion_spec.rb
@@ -5,7 +5,9 @@ describe 'the `completion` suggestion strategy' do
session.
run_command('autoload compinit && compinit').
run_command('_foo() { compadd bar; compadd bat }').
- run_command('compdef _foo baz')
+ run_command('_num() { compadd two; compadd three }').
+ run_command('compdef _foo baz').
+ run_command('compdef _num one')
end
end
@@ -37,6 +39,21 @@ describe 'the `completion` suggestion strategy' do
end
end
+ context 'when ZSH_AUTOSUGGEST_COMPLETION_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=completion', 'ZSH_AUTOSUGGEST_COMPLETION_IGNORE="one *"'] }
+
+ it 'makes suggestions when the buffer does not match the pattern' do
+ session.send_string('baz ')
+ wait_for { session.content }.to eq('baz bar')
+ end
+
+ it 'does not make suggestions when the buffer matches the pattern' do
+ session.send_string('one t')
+ sleep 1
+ expect(session.content).to eq('one t')
+ end
+ end
+
context 'when async mode is enabled' do
let(:options) { ['ZSH_AUTOSUGGEST_USE_ASYNC=true', 'ZSH_AUTOSUGGEST_STRATEGY=completion'] }
diff --git a/spec/strategies/history_spec.rb b/spec/strategies/history_spec.rb
index f8ae526..eee8efd 100644
--- a/spec/strategies/history_spec.rb
+++ b/spec/strategies/history_spec.rb
@@ -8,5 +8,16 @@ describe 'the `history` suggestion strategy' do
end
end
+ context 'when ZSH_AUTOSUGGEST_HISTORY_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_HISTORY_IGNORE="* bar"'] }
+
+ it 'does not make suggestions that match the pattern' do
+ with_history('ls foo', 'ls bar', 'echo baz') do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls foo')
+ end
+ end
+ end
+
include_examples 'special characters'
end
diff --git a/spec/strategies/match_prev_cmd_spec.rb b/spec/strategies/match_prev_cmd_spec.rb
index 5a143b8..c435f16 100644
--- a/spec/strategies/match_prev_cmd_spec.rb
+++ b/spec/strategies/match_prev_cmd_spec.rb
@@ -3,19 +3,32 @@ require 'strategies/special_characters_helper'
describe 'the `match_prev_cmd` strategy' do
let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=match_prev_cmd'] }
+ let(:history) { [
+ 'echo what',
+ 'ls foo',
+ 'echo what',
+ 'ls bar',
+ 'ls baz',
+ 'echo what'
+ ] }
+
it 'suggests the last matching history entry after the previous command' do
- with_history(
- 'echo what',
- 'ls foo',
- 'echo what',
- 'ls bar',
- 'ls baz',
- 'echo what'
- ) do
+ with_history(*history) do
session.send_string('ls')
wait_for { session.content }.to eq('ls bar')
end
end
+ context 'when ZSH_AUTOSUGGEST_HISTORY_IGNORE is set to a pattern' do
+ let(:options) { ['ZSH_AUTOSUGGEST_STRATEGY=match_prev_cmd', 'ZSH_AUTOSUGGEST_HISTORY_IGNORE="* bar"'] }
+
+ it 'does not make suggestions that match the pattern' do
+ with_history(*history) do
+ session.send_string('ls')
+ wait_for { session.content }.to eq('ls foo')
+ end
+ end
+ end
+
include_examples 'special characters'
end
diff --git a/spec/strategies/special_characters_helper.rb b/spec/strategies/special_characters_helper.rb
index 8771861..eb1f0cd 100644
--- a/spec/strategies/special_characters_helper.rb
+++ b/spec/strategies/special_characters_helper.rb
@@ -1,58 +1,71 @@
shared_examples 'special characters' do
- describe 'a special character in the buffer' do
- it 'should be treated like any other character' do
+ describe 'a special character in the buffer should be treated like any other character' do
+ it 'asterisk' do
with_history('echo "hello*"', 'echo "hello."') do
session.send_string('echo "hello*')
wait_for { session.content }.to eq('echo "hello*"')
end
+ end
+ it 'question mark' do
with_history('echo "hello?"', 'echo "hello."') do
session.send_string('echo "hello?')
wait_for { session.content }.to eq('echo "hello?"')
end
+ end
+ it 'backslash' do
with_history('echo "hello\nworld"') do
session.send_string('echo "hello\\')
wait_for { session.content }.to eq('echo "hello\nworld"')
end
+ end
+ it 'double backslash' do
with_history('echo "\\\\"') do
session.send_string('echo "\\\\')
wait_for { session.content }.to eq('echo "\\\\"')
end
+ end
+ it 'tilde' do
with_history('echo ~/foo') do
session.send_string('echo ~')
wait_for { session.content }.to eq('echo ~/foo')
end
+ end
+ it 'parentheses' do
with_history('echo "$(ls foo)"') do
session.send_string('echo "$(')
wait_for { session.content }.to eq('echo "$(ls foo)"')
end
+ end
+ it 'square bracket' do
with_history('echo "$history[123]"') do
session.send_string('echo "$history[')
wait_for { session.content }.to eq('echo "$history[123]"')
session.send_string('123]')
wait_for { session.content }.to eq('echo "$history[123]"')
end
+ end
+ it 'octothorpe' do
with_history('echo "#yolo"') do
session.send_string('echo "#')
wait_for { session.content }.to eq('echo "#yolo"')
end
+ end
- with_history('echo "#foo"', 'echo $#abc') do
- session.send_string('echo "#')
- wait_for { session.content }.to eq('echo "#foo"')
- end
-
+ it 'caret' do
with_history('echo "^A"', 'echo "^B"') do
session.send_string('echo "^A')
wait_for { session.content }.to eq('echo "^A"')
end
+ end
+ it 'dash' do
with_history('-foo() {}') do
session.send_string('-')
wait_for { session.content }.to eq('-foo() {}')
diff --git a/src/async.zsh b/src/async.zsh
index 4314e8c..e179734 100644
--- a/src/async.zsh
+++ b/src/async.zsh
@@ -11,7 +11,7 @@ _zsh_autosuggest_async_request() {
# If we've got a pending request, cancel it
if [[ -n "$_ZSH_AUTOSUGGEST_ASYNC_FD" ]] && { true <&$_ZSH_AUTOSUGGEST_ASYNC_FD } 2>/dev/null; then
# Close the file descriptor and remove the handler
- exec {_ZSH_AUTOSUGGEST_ASYNC_FD}<&-
+ builtin exec {_ZSH_AUTOSUGGEST_ASYNC_FD}<&-
zle -F $_ZSH_AUTOSUGGEST_ASYNC_FD
# We won't know the pid unless the user has zsh/system module installed
@@ -32,7 +32,7 @@ _zsh_autosuggest_async_request() {
fi
# Fork a process to fetch a suggestion and open a pipe to read from it
- exec {_ZSH_AUTOSUGGEST_ASYNC_FD}< <(
+ builtin exec {_ZSH_AUTOSUGGEST_ASYNC_FD}< <(
# Tell parent process our pid
echo $sysparams[pid]
@@ -44,7 +44,8 @@ _zsh_autosuggest_async_request() {
# There's a weird bug here where ^C stops working unless we force a fork
# See https://github.com/zsh-users/zsh-autosuggestions/issues/364
- command true
+ autoload -Uz is-at-least
+ is-at-least 5.8 || command true
# Read the pid from the child process
read _ZSH_AUTOSUGGEST_CHILD_PID <&$_ZSH_AUTOSUGGEST_ASYNC_FD
@@ -67,9 +68,10 @@ _zsh_autosuggest_async_response() {
zle autosuggest-suggest -- "$suggestion"
# Close the fd
- exec {1}<&-
+ builtin exec {1}<&-
fi
# Always remove the handler
zle -F "$1"
+ _ZSH_AUTOSUGGEST_ASYNC_FD=
}
diff --git a/src/bind.zsh b/src/bind.zsh
index fc2da9e..1dde137 100644
--- a/src/bind.zsh
+++ b/src/bind.zsh
@@ -69,7 +69,7 @@ _zsh_autosuggest_bind_widgets() {
ignore_widgets=(
.\*
_\*
- autosuggest-\*
+ ${_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS/#/autosuggest-}
$ZSH_AUTOSUGGEST_ORIGINAL_WIDGET_PREFIX\*
$ZSH_AUTOSUGGEST_IGNORE_WIDGETS
)
diff --git a/src/config.zsh b/src/config.zsh
index 5a0ebd8..32d32b2 100644
--- a/src/config.zsh
+++ b/src/config.zsh
@@ -28,6 +28,8 @@ typeset -g ZSH_AUTOSUGGEST_ORIGINAL_WIDGET_PREFIX=autosuggest-orig-
history-search-backward
history-beginning-search-forward
history-beginning-search-backward
+ history-beginning-search-forward-end
+ history-beginning-search-backward-end
history-substring-search-up
history-substring-search-down
up-line-or-beginning-search
diff --git a/src/start.zsh b/src/start.zsh
index 5991039..5d4ee52 100644
--- a/src/start.zsh
+++ b/src/start.zsh
@@ -18,6 +18,16 @@ _zsh_autosuggest_start() {
_zsh_autosuggest_bind_widgets
}
+# Mark for auto-loading the functions that we use
+autoload -Uz add-zsh-hook is-at-least
+
+# Automatically enable asynchronous mode in newer versions of zsh. Disable for
+# older versions because there is a bug when using async mode where ^C does not
+# work immediately after fetching a suggestion.
+# See https://github.com/zsh-users/zsh-autosuggestions/issues/364
+if is-at-least 5.0.8; then
+ typeset -g ZSH_AUTOSUGGEST_USE_ASYNC=
+fi
+
# Start the autosuggestion widgets on the next precmd
-autoload -Uz add-zsh-hook
add-zsh-hook precmd _zsh_autosuggest_start
diff --git a/src/strategies/completion.zsh b/src/strategies/completion.zsh
index dfbf6eb..e2d114c 100644
--- a/src/strategies/completion.zsh
+++ b/src/strategies/completion.zsh
@@ -45,8 +45,6 @@ _zsh_autosuggest_capture_completion_widget() {
zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget
_zsh_autosuggest_capture_setup() {
- autoload -Uz is-at-least
-
# There is a bug in zpty module in older zsh versions by which a
# zpty that exits will kill all zpty processes that were forked
# before it. Here we set up a zsh exit hook to SIGKILL the zpty
@@ -96,6 +94,12 @@ _zsh_autosuggest_capture_completion_async() {
}
_zsh_autosuggest_strategy_completion() {
+ # Reset options to defaults and enable LOCAL_OPTIONS
+ emulate -L zsh
+
+ # Enable extended glob for completion ignore pattern
+ setopt EXTENDED_GLOB
+
typeset -g suggestion
local line REPLY
@@ -105,6 +109,9 @@ _zsh_autosuggest_strategy_completion() {
# Exit if we don't have zpty
zmodload zsh/zpty 2>/dev/null || return
+ # Exit if our search string matches the ignore pattern
+ [[ -n "$ZSH_AUTOSUGGEST_COMPLETION_IGNORE" ]] && [[ "$1" == $~ZSH_AUTOSUGGEST_COMPLETION_IGNORE ]] && return
+
# Zle will be inactive if we are in async mode
if zle; then
zpty $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME _zsh_autosuggest_capture_completion_sync
@@ -122,7 +129,7 @@ _zsh_autosuggest_strategy_completion() {
# versions of zsh (older than 5.3), we sometimes get extra bytes after
# the second null byte, so trim those off the end.
# See http://www.zsh.org/mla/workers/2015/msg03290.html
- suggestion="${${line#*$'\0'}%$'\0'*}"
+ suggestion="${${(@0)line}[2]}"
} always {
# Destroy the pty
zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME
diff --git a/src/strategies/history.zsh b/src/strategies/history.zsh
index a2755a5..0672a13 100644
--- a/src/strategies/history.zsh
+++ b/src/strategies/history.zsh
@@ -10,7 +10,7 @@ _zsh_autosuggest_strategy_history() {
# Reset options to defaults and enable LOCAL_OPTIONS
emulate -L zsh
- # Enable globbing flags so that we can use (#m)
+ # Enable globbing flags so that we can use (#m) and (x~y) glob operator
setopt EXTENDED_GLOB
# Escape backslashes and all of the glob operators so we can use
@@ -19,7 +19,14 @@ _zsh_autosuggest_strategy_history() {
# TODO: Use (b) flag when we can drop support for zsh older than v5.0.8
local prefix="${1//(#m)[\\*?[\]<>()|^~#]/\\$MATCH}"
- # Get the history items that match
+ # Get the history items that match the prefix, excluding those that match
+ # the ignore pattern
+ local pattern="$prefix*"
+ if [[ -n $ZSH_AUTOSUGGEST_HISTORY_IGNORE ]]; then
+ pattern="($pattern)~($ZSH_AUTOSUGGEST_HISTORY_IGNORE)"
+ fi
+
+ # Give the first history item matching the pattern as the suggestion
# - (r) subscript flag makes the pattern match on values
- typeset -g suggestion="${history[(r)${prefix}*]}"
+ typeset -g suggestion="${history[(r)$pattern]}"
}
diff --git a/src/strategies/match_prev_cmd.zsh b/src/strategies/match_prev_cmd.zsh
index f76d3c1..b709783 100644
--- a/src/strategies/match_prev_cmd.zsh
+++ b/src/strategies/match_prev_cmd.zsh
@@ -24,16 +24,23 @@ _zsh_autosuggest_strategy_match_prev_cmd() {
# Reset options to defaults and enable LOCAL_OPTIONS
emulate -L zsh
- # Enable globbing flags so that we can use (#m)
+ # Enable globbing flags so that we can use (#m) and (x~y) glob operator
setopt EXTENDED_GLOB
# TODO: Use (b) flag when we can drop support for zsh older than v5.0.8
local prefix="${1//(#m)[\\*?[\]<>()|^~#]/\\$MATCH}"
+ # Get the history items that match the prefix, excluding those that match
+ # the ignore pattern
+ local pattern="$prefix*"
+ if [[ -n $ZSH_AUTOSUGGEST_HISTORY_IGNORE ]]; then
+ pattern="($pattern)~($ZSH_AUTOSUGGEST_HISTORY_IGNORE)"
+ fi
+
# Get all history event numbers that correspond to history
- # entries that match pattern $prefix*
+ # entries that match the pattern
local history_match_keys
- history_match_keys=(${(k)history[(R)$prefix*]})
+ history_match_keys=(${(k)history[(R)$~pattern]})
# By default we use the first history number (most recent history entry)
local histkey="${history_match_keys[1]}"
diff --git a/src/widgets.zsh b/src/widgets.zsh
index 242d502..7562897 100644
--- a/src/widgets.zsh
+++ b/src/widgets.zsh
@@ -20,7 +20,7 @@ _zsh_autosuggest_enable() {
# Toggle suggestions (enable/disable)
_zsh_autosuggest_toggle() {
- if [[ -n "${_ZSH_AUTOSUGGEST_DISABLED+x}" ]]; then
+ if (( ${+_ZSH_AUTOSUGGEST_DISABLED} )); then
_zsh_autosuggest_enable
else
_zsh_autosuggest_disable
@@ -30,7 +30,7 @@ _zsh_autosuggest_toggle() {
# Clear the suggestion
_zsh_autosuggest_clear() {
# Remove the suggestion
- unset POSTDISPLAY
+ POSTDISPLAY=
_zsh_autosuggest_invoke_original_widget $@
}
@@ -47,7 +47,7 @@ _zsh_autosuggest_modify() {
local orig_postdisplay="$POSTDISPLAY"
# Clear suggestion while waiting for next one
- unset POSTDISPLAY
+ POSTDISPLAY=
# Original widget may modify the buffer
_zsh_autosuggest_invoke_original_widget $@
@@ -56,30 +56,19 @@ _zsh_autosuggest_modify() {
emulate -L zsh
# Don't fetch a new suggestion if there's more input to be read immediately
- if (( $PENDING > 0 )) || (( $KEYS_QUEUED_COUNT > 0 )); then
+ if (( $PENDING > 0 || $KEYS_QUEUED_COUNT > 0 )); then
POSTDISPLAY="$orig_postdisplay"
return $retval
fi
- # Optimize if manually typing in the suggestion
- if (( $#BUFFER > $#orig_buffer )); then
- local added=${BUFFER#$orig_buffer}
-
- # If the string added matches the beginning of the postdisplay
- if [[ "$added" = "${orig_postdisplay:0:$#added}" ]]; then
- POSTDISPLAY="${orig_postdisplay:$#added}"
- return $retval
- fi
- fi
-
- # Don't fetch a new suggestion if the buffer hasn't changed
- if [[ "$BUFFER" = "$orig_buffer" ]]; then
- POSTDISPLAY="$orig_postdisplay"
+ # Optimize if manually typing in the suggestion or if buffer hasn't changed
+ if [[ "$BUFFER" = "$orig_buffer"* && "$orig_postdisplay" = "${BUFFER:$#orig_buffer}"* ]]; then
+ POSTDISPLAY="${orig_postdisplay:$(($#BUFFER - $#orig_buffer))}"
return $retval
fi
# Bail out if suggestions are disabled
- if [[ -n "${_ZSH_AUTOSUGGEST_DISABLED+x}" ]]; then
+ if (( ${+_ZSH_AUTOSUGGEST_DISABLED} )); then
return $?
fi
@@ -113,13 +102,13 @@ _zsh_autosuggest_suggest() {
if [[ -n "$suggestion" ]] && (( $#BUFFER )); then
POSTDISPLAY="${suggestion#$BUFFER}"
else
- unset POSTDISPLAY
+ POSTDISPLAY=
fi
}
# Accept the entire suggestion
_zsh_autosuggest_accept() {
- local -i max_cursor_pos=$#BUFFER
+ local -i retval max_cursor_pos=$#BUFFER
# When vicmd keymap is active, the cursor can't move all the way
# to the end of the buffer
@@ -127,23 +116,33 @@ _zsh_autosuggest_accept() {
max_cursor_pos=$((max_cursor_pos - 1))
fi
- # Only accept if the cursor is at the end of the buffer
- if [[ $CURSOR = $max_cursor_pos ]]; then
- # Add the suggestion to the buffer
- BUFFER="$BUFFER$POSTDISPLAY"
-
- # Remove the suggestion
- unset POSTDISPLAY
-
- # Move the cursor to the end of the buffer
- if [[ "$KEYMAP" = "vicmd" ]]; then
- CURSOR=$(($#BUFFER - 1))
- else
- CURSOR=$#BUFFER
- fi
+ # If we're not in a valid state to accept a suggestion, just run the
+ # original widget and bail out
+ if (( $CURSOR != $max_cursor_pos || !$#POSTDISPLAY )); then
+ _zsh_autosuggest_invoke_original_widget $@
+ return
fi
+ # Only accept if the cursor is at the end of the buffer
+ # Add the suggestion to the buffer
+ BUFFER="$BUFFER$POSTDISPLAY"
+
+ # Remove the suggestion
+ POSTDISPLAY=
+
+ # Run the original widget before manually moving the cursor so that the
+ # cursor movement doesn't make the widget do something unexpected
_zsh_autosuggest_invoke_original_widget $@
+ retval=$?
+
+ # Move the cursor to the end of the buffer
+ if [[ "$KEYMAP" = "vicmd" ]]; then
+ CURSOR=$(($#BUFFER - 1))
+ else
+ CURSOR=$#BUFFER
+ fi
+
+ return $retval
}
# Accept the entire suggestion and execute it
@@ -152,7 +151,7 @@ _zsh_autosuggest_execute() {
BUFFER="$BUFFER$POSTDISPLAY"
# Remove the suggestion
- unset POSTDISPLAY
+ POSTDISPLAY=
# Call the original `accept-line` to handle syntax highlighting or
# other potential custom behavior
@@ -195,8 +194,21 @@ _zsh_autosuggest_partial_accept() {
}
() {
+ typeset -ga _ZSH_AUTOSUGGEST_BUILTIN_ACTIONS
+
+ _ZSH_AUTOSUGGEST_BUILTIN_ACTIONS=(
+ clear
+ fetch
+ suggest
+ accept
+ execute
+ enable
+ disable
+ toggle
+ )
+
local action
- for action in clear modify fetch suggest accept partial_accept execute enable disable toggle; do
+ for action in $_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS modify partial_accept; do
eval "_zsh_autosuggest_widget_$action() {
local -i retval
@@ -213,12 +225,7 @@ _zsh_autosuggest_partial_accept() {
}"
done
- zle -N autosuggest-fetch _zsh_autosuggest_widget_fetch
- zle -N autosuggest-suggest _zsh_autosuggest_widget_suggest
- zle -N autosuggest-accept _zsh_autosuggest_widget_accept
- zle -N autosuggest-clear _zsh_autosuggest_widget_clear
- zle -N autosuggest-execute _zsh_autosuggest_widget_execute
- zle -N autosuggest-enable _zsh_autosuggest_widget_enable
- zle -N autosuggest-disable _zsh_autosuggest_widget_disable
- zle -N autosuggest-toggle _zsh_autosuggest_widget_toggle
+ for action in $_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS; do
+ zle -N autosuggest-$action _zsh_autosuggest_widget_$action
+ done
}
diff --git a/zsh-autosuggestions.zsh b/zsh-autosuggestions.zsh
index 872b647..e780225 100644
--- a/zsh-autosuggestions.zsh
+++ b/zsh-autosuggestions.zsh
@@ -1,8 +1,8 @@
# Fish-like fast/unobtrusive autosuggestions for zsh.
# https://github.com/zsh-users/zsh-autosuggestions
-# v0.6.3
+# v0.7.1
# Copyright (c) 2013 Thiago de Arruda
-# Copyright (c) 2016-2019 Eric Freese
+# Copyright (c) 2016-2021 Eric Freese
#
# Permission is hereby granted, free of charge, to any person
# obtaining a copy of this software and associated documentation
@@ -54,6 +54,8 @@ typeset -g ZSH_AUTOSUGGEST_ORIGINAL_WIDGET_PREFIX=autosuggest-orig-
history-search-backward
history-beginning-search-forward
history-beginning-search-backward
+ history-beginning-search-forward-end
+ history-beginning-search-backward-end
history-substring-search-up
history-substring-search-down
up-line-or-beginning-search
@@ -199,7 +201,7 @@ _zsh_autosuggest_bind_widgets() {
ignore_widgets=(
.\*
_\*
- autosuggest-\*
+ ${_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS/#/autosuggest-}
$ZSH_AUTOSUGGEST_ORIGINAL_WIDGET_PREFIX\*
$ZSH_AUTOSUGGEST_IGNORE_WIDGETS
)
@@ -282,7 +284,7 @@ _zsh_autosuggest_enable() {
# Toggle suggestions (enable/disable)
_zsh_autosuggest_toggle() {
- if [[ -n "${_ZSH_AUTOSUGGEST_DISABLED+x}" ]]; then
+ if (( ${+_ZSH_AUTOSUGGEST_DISABLED} )); then
_zsh_autosuggest_enable
else
_zsh_autosuggest_disable
@@ -292,7 +294,7 @@ _zsh_autosuggest_toggle() {
# Clear the suggestion
_zsh_autosuggest_clear() {
# Remove the suggestion
- unset POSTDISPLAY
+ POSTDISPLAY=
_zsh_autosuggest_invoke_original_widget $@
}
@@ -309,7 +311,7 @@ _zsh_autosuggest_modify() {
local orig_postdisplay="$POSTDISPLAY"
# Clear suggestion while waiting for next one
- unset POSTDISPLAY
+ POSTDISPLAY=
# Original widget may modify the buffer
_zsh_autosuggest_invoke_original_widget $@
@@ -318,30 +320,19 @@ _zsh_autosuggest_modify() {
emulate -L zsh
# Don't fetch a new suggestion if there's more input to be read immediately
- if (( $PENDING > 0 )) || (( $KEYS_QUEUED_COUNT > 0 )); then
+ if (( $PENDING > 0 || $KEYS_QUEUED_COUNT > 0 )); then
POSTDISPLAY="$orig_postdisplay"
return $retval
fi
- # Optimize if manually typing in the suggestion
- if (( $#BUFFER > $#orig_buffer )); then
- local added=${BUFFER#$orig_buffer}
-
- # If the string added matches the beginning of the postdisplay
- if [[ "$added" = "${orig_postdisplay:0:$#added}" ]]; then
- POSTDISPLAY="${orig_postdisplay:$#added}"
- return $retval
- fi
- fi
-
- # Don't fetch a new suggestion if the buffer hasn't changed
- if [[ "$BUFFER" = "$orig_buffer" ]]; then
- POSTDISPLAY="$orig_postdisplay"
+ # Optimize if manually typing in the suggestion or if buffer hasn't changed
+ if [[ "$BUFFER" = "$orig_buffer"* && "$orig_postdisplay" = "${BUFFER:$#orig_buffer}"* ]]; then
+ POSTDISPLAY="${orig_postdisplay:$(($#BUFFER - $#orig_buffer))}"
return $retval
fi
# Bail out if suggestions are disabled
- if [[ -n "${_ZSH_AUTOSUGGEST_DISABLED+x}" ]]; then
+ if (( ${+_ZSH_AUTOSUGGEST_DISABLED} )); then
return $?
fi
@@ -375,13 +366,13 @@ _zsh_autosuggest_suggest() {
if [[ -n "$suggestion" ]] && (( $#BUFFER )); then
POSTDISPLAY="${suggestion#$BUFFER}"
else
- unset POSTDISPLAY
+ POSTDISPLAY=
fi
}
# Accept the entire suggestion
_zsh_autosuggest_accept() {
- local -i max_cursor_pos=$#BUFFER
+ local -i retval max_cursor_pos=$#BUFFER
# When vicmd keymap is active, the cursor can't move all the way
# to the end of the buffer
@@ -389,23 +380,33 @@ _zsh_autosuggest_accept() {
max_cursor_pos=$((max_cursor_pos - 1))
fi
- # Only accept if the cursor is at the end of the buffer
- if [[ $CURSOR = $max_cursor_pos ]]; then
- # Add the suggestion to the buffer
- BUFFER="$BUFFER$POSTDISPLAY"
-
- # Remove the suggestion
- unset POSTDISPLAY
-
- # Move the cursor to the end of the buffer
- if [[ "$KEYMAP" = "vicmd" ]]; then
- CURSOR=$(($#BUFFER - 1))
- else
- CURSOR=$#BUFFER
- fi
+ # If we're not in a valid state to accept a suggestion, just run the
+ # original widget and bail out
+ if (( $CURSOR != $max_cursor_pos || !$#POSTDISPLAY )); then
+ _zsh_autosuggest_invoke_original_widget $@
+ return
fi
+ # Only accept if the cursor is at the end of the buffer
+ # Add the suggestion to the buffer
+ BUFFER="$BUFFER$POSTDISPLAY"
+
+ # Remove the suggestion
+ POSTDISPLAY=
+
+ # Run the original widget before manually moving the cursor so that the
+ # cursor movement doesn't make the widget do something unexpected
_zsh_autosuggest_invoke_original_widget $@
+ retval=$?
+
+ # Move the cursor to the end of the buffer
+ if [[ "$KEYMAP" = "vicmd" ]]; then
+ CURSOR=$(($#BUFFER - 1))
+ else
+ CURSOR=$#BUFFER
+ fi
+
+ return $retval
}
# Accept the entire suggestion and execute it
@@ -414,7 +415,7 @@ _zsh_autosuggest_execute() {
BUFFER="$BUFFER$POSTDISPLAY"
# Remove the suggestion
- unset POSTDISPLAY
+ POSTDISPLAY=
# Call the original `accept-line` to handle syntax highlighting or
# other potential custom behavior
@@ -457,8 +458,21 @@ _zsh_autosuggest_partial_accept() {
}
() {
+ typeset -ga _ZSH_AUTOSUGGEST_BUILTIN_ACTIONS
+
+ _ZSH_AUTOSUGGEST_BUILTIN_ACTIONS=(
+ clear
+ fetch
+ suggest
+ accept
+ execute
+ enable
+ disable
+ toggle
+ )
+
local action
- for action in clear modify fetch suggest accept partial_accept execute enable disable toggle; do
+ for action in $_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS modify partial_accept; do
eval "_zsh_autosuggest_widget_$action() {
local -i retval
@@ -475,14 +489,9 @@ _zsh_autosuggest_partial_accept() {
}"
done
- zle -N autosuggest-fetch _zsh_autosuggest_widget_fetch
- zle -N autosuggest-suggest _zsh_autosuggest_widget_suggest
- zle -N autosuggest-accept _zsh_autosuggest_widget_accept
- zle -N autosuggest-clear _zsh_autosuggest_widget_clear
- zle -N autosuggest-execute _zsh_autosuggest_widget_execute
- zle -N autosuggest-enable _zsh_autosuggest_widget_enable
- zle -N autosuggest-disable _zsh_autosuggest_widget_disable
- zle -N autosuggest-toggle _zsh_autosuggest_widget_toggle
+ for action in $_ZSH_AUTOSUGGEST_BUILTIN_ACTIONS; do
+ zle -N autosuggest-$action _zsh_autosuggest_widget_$action
+ done
}
#--------------------------------------------------------------------#
@@ -531,8 +540,6 @@ _zsh_autosuggest_capture_completion_widget() {
zle -N autosuggest-capture-completion _zsh_autosuggest_capture_completion_widget
_zsh_autosuggest_capture_setup() {
- autoload -Uz is-at-least
-
# There is a bug in zpty module in older zsh versions by which a
# zpty that exits will kill all zpty processes that were forked
# before it. Here we set up a zsh exit hook to SIGKILL the zpty
@@ -582,6 +589,12 @@ _zsh_autosuggest_capture_completion_async() {
}
_zsh_autosuggest_strategy_completion() {
+ # Reset options to defaults and enable LOCAL_OPTIONS
+ emulate -L zsh
+
+ # Enable extended glob for completion ignore pattern
+ setopt EXTENDED_GLOB
+
typeset -g suggestion
local line REPLY
@@ -591,6 +604,9 @@ _zsh_autosuggest_strategy_completion() {
# Exit if we don't have zpty
zmodload zsh/zpty 2>/dev/null || return
+ # Exit if our search string matches the ignore pattern
+ [[ -n "$ZSH_AUTOSUGGEST_COMPLETION_IGNORE" ]] && [[ "$1" == $~ZSH_AUTOSUGGEST_COMPLETION_IGNORE ]] && return
+
# Zle will be inactive if we are in async mode
if zle; then
zpty $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME _zsh_autosuggest_capture_completion_sync
@@ -608,7 +624,7 @@ _zsh_autosuggest_strategy_completion() {
# versions of zsh (older than 5.3), we sometimes get extra bytes after
# the second null byte, so trim those off the end.
# See http://www.zsh.org/mla/workers/2015/msg03290.html
- suggestion="${${line#*$'\0'}%$'\0'*}"
+ suggestion="${${(@0)line}[2]}"
} always {
# Destroy the pty
zpty -d $ZSH_AUTOSUGGEST_COMPLETIONS_PTY_NAME
@@ -626,7 +642,7 @@ _zsh_autosuggest_strategy_history() {
# Reset options to defaults and enable LOCAL_OPTIONS
emulate -L zsh
- # Enable globbing flags so that we can use (#m)
+ # Enable globbing flags so that we can use (#m) and (x~y) glob operator
setopt EXTENDED_GLOB
# Escape backslashes and all of the glob operators so we can use
@@ -635,9 +651,16 @@ _zsh_autosuggest_strategy_history() {
# TODO: Use (b) flag when we can drop support for zsh older than v5.0.8
local prefix="${1//(#m)[\\*?[\]<>()|^~#]/\\$MATCH}"
- # Get the history items that match
+ # Get the history items that match the prefix, excluding those that match
+ # the ignore pattern
+ local pattern="$prefix*"
+ if [[ -n $ZSH_AUTOSUGGEST_HISTORY_IGNORE ]]; then
+ pattern="($pattern)~($ZSH_AUTOSUGGEST_HISTORY_IGNORE)"
+ fi
+
+ # Give the first history item matching the pattern as the suggestion
# - (r) subscript flag makes the pattern match on values
- typeset -g suggestion="${history[(r)${prefix}*]}"
+ typeset -g suggestion="${history[(r)$pattern]}"
}
#--------------------------------------------------------------------#
@@ -665,16 +688,23 @@ _zsh_autosuggest_strategy_match_prev_cmd() {
# Reset options to defaults and enable LOCAL_OPTIONS
emulate -L zsh
- # Enable globbing flags so that we can use (#m)
+ # Enable globbing flags so that we can use (#m) and (x~y) glob operator
setopt EXTENDED_GLOB
# TODO: Use (b) flag when we can drop support for zsh older than v5.0.8
local prefix="${1//(#m)[\\*?[\]<>()|^~#]/\\$MATCH}"
+ # Get the history items that match the prefix, excluding those that match
+ # the ignore pattern
+ local pattern="$prefix*"
+ if [[ -n $ZSH_AUTOSUGGEST_HISTORY_IGNORE ]]; then
+ pattern="($pattern)~($ZSH_AUTOSUGGEST_HISTORY_IGNORE)"
+ fi
+
# Get all history event numbers that correspond to history
- # entries that match pattern $prefix*
+ # entries that match the pattern
local history_match_keys
- history_match_keys=(${(k)history[(R)$prefix*]})
+ history_match_keys=(${(k)history[(R)$~pattern]})
# By default we use the first history number (most recent history entry)
local histkey="${history_match_keys[1]}"
@@ -738,7 +768,7 @@ _zsh_autosuggest_async_request() {
# If we've got a pending request, cancel it
if [[ -n "$_ZSH_AUTOSUGGEST_ASYNC_FD" ]] && { true <&$_ZSH_AUTOSUGGEST_ASYNC_FD } 2>/dev/null; then
# Close the file descriptor and remove the handler
- exec {_ZSH_AUTOSUGGEST_ASYNC_FD}<&-
+ builtin exec {_ZSH_AUTOSUGGEST_ASYNC_FD}<&-
zle -F $_ZSH_AUTOSUGGEST_ASYNC_FD
# We won't know the pid unless the user has zsh/system module installed
@@ -759,7 +789,7 @@ _zsh_autosuggest_async_request() {
fi
# Fork a process to fetch a suggestion and open a pipe to read from it
- exec {_ZSH_AUTOSUGGEST_ASYNC_FD}< <(
+ builtin exec {_ZSH_AUTOSUGGEST_ASYNC_FD}< <(
# Tell parent process our pid
echo $sysparams[pid]
@@ -771,7 +801,8 @@ _zsh_autosuggest_async_request() {
# There's a weird bug here where ^C stops working unless we force a fork
# See https://github.com/zsh-users/zsh-autosuggestions/issues/364
- command true
+ autoload -Uz is-at-least
+ is-at-least 5.8 || command true
# Read the pid from the child process
read _ZSH_AUTOSUGGEST_CHILD_PID <&$_ZSH_AUTOSUGGEST_ASYNC_FD
@@ -794,11 +825,12 @@ _zsh_autosuggest_async_response() {
zle autosuggest-suggest -- "$suggestion"
# Close the fd
- exec {1}<&-
+ builtin exec {1}<&-
fi
# Always remove the handler
zle -F "$1"
+ _ZSH_AUTOSUGGEST_ASYNC_FD=
}
#--------------------------------------------------------------------#
@@ -820,6 +852,16 @@ _zsh_autosuggest_start() {
_zsh_autosuggest_bind_widgets
}
+# Mark for auto-loading the functions that we use
+autoload -Uz add-zsh-hook is-at-least
+
+# Automatically enable asynchronous mode in newer versions of zsh. Disable for
+# older versions because there is a bug when using async mode where ^C does not
+# work immediately after fetching a suggestion.
+# See https://github.com/zsh-users/zsh-autosuggestions/issues/364
+if is-at-least 5.0.8; then
+ typeset -g ZSH_AUTOSUGGEST_USE_ASYNC=
+fi
+
# Start the autosuggestion widgets on the next precmd
-autoload -Uz add-zsh-hook
add-zsh-hook precmd _zsh_autosuggest_start