An open API service indexing awesome lists of open source software.

https://github.com/jcs-elpa/company-fuzzy

Fuzzy matching for `company-mode'
https://github.com/jcs-elpa/company-fuzzy

auto-complete emacs ycmd

Last synced: 6 months ago
JSON representation

Fuzzy matching for `company-mode'

Awesome Lists containing this project

README

          

[![License: GPL v3](https://img.shields.io/badge/License-GPL%20v3-blue.svg)](https://www.gnu.org/licenses/gpl-3.0)
[![JCS-ELPA](https://raw.githubusercontent.com/jcs-emacs/badges/master/elpa/v/company-fuzzy.svg)](https://jcs-emacs.github.io/jcs-elpa/#/company-fuzzy)
[![MELPA](https://melpa.org/packages/company-fuzzy-badge.svg)](https://melpa.org/#/company-fuzzy)
[![MELPA Stable](https://stable.melpa.org/packages/company-fuzzy-badge.svg)](https://stable.melpa.org/#/company-fuzzy)

# company-fuzzy
> Fuzzy matching for `company-mode'.

[![CI](https://github.com/jcs-elpa/company-fuzzy/actions/workflows/test.yml/badge.svg)](https://github.com/jcs-elpa/company-fuzzy/actions/workflows/test.yml)



Pure `elisp` fuzzy completion for `company-mode`. This plugin search through
all the buffer local `company-backends` and fuzzy search all candidates.

## ๐Ÿ† Features

* *Work across all backends* - Any backend that gives list of string should work.
* *Only uses native `elisp` code* - I personally don't prefer any external
program unless is necessary.
* *Combined all backends to one backend* - Opposite to [company-try-hard](https://github.com/Wilfred/company-try-hard),
hence all possible candidates will be shown in the auto-complete menu.

## ๐Ÿงช Differences from other alternatives

* [company-ycmd](https://github.com/abingham/emacs-ycmd)
* Uses [ycmd](https://github.com/Valloric/ycmd) as backend to provide functionalities.
* Quite hard to config properly.
* [company-flx](https://github.com/PythonNut/company-flx)
* Uses library [flx](https://github.com/lewang/flx). (Optional for us)
* Only works with `capf` backend currently.

## ๐Ÿ’พ Quickstart

```el
(use-package company-fuzzy
:hook (company-mode . company-fuzzy-mode)
:init
(setq company-fuzzy-sorting-backend 'flx
company-fuzzy-reset-selection t
company-fuzzy-prefix-on-top nil
company-fuzzy-trigger-symbols '("." "->" "<" "\"" "'" "@")))
```

## ๐Ÿ”ง Usage

You can enable it globally by adding this line to your config
```el
(global-company-fuzzy-mode 1)
```
Or you can just enable it in any specific buffer/mode you want.
```el
(company-fuzzy-mode 1)
```

Make sure you have called either of these functions after all
`company-backends` are set and config properly. Because this plugin will
replace all backends to this minor mode specific backend (basically take all
backends away, so this mode could combine all sources and do the fuzzy work).

### ๐Ÿ” Sorting/Scoring backend

There are multiple sorting algorithms for auto-completion. You can choose your
backend by customizing the `company-fuzzy-sorting-backend` variable like this.

```el
(setq company-fuzzy-sorting-backend 'alphabetic)
```

Currently supports these values,

- *`none`* - Gives you the raw result.
- *`alphabetic`* - Sort in the alphabetic order. (VSCode)
- *`flex`* - Use library [flex](https://github.com/jcs-elpa/flex) as matching engine.
- *`flx`* - Use library [flx](https://github.com/lewang/flx) as matching engine. (Sublime Text)
- *`flx-rs`* - Use library [flx-rs](https://github.com/jcs-elpa/flx-rs) as matching engine. (Sublime Text)
- *`flxy`* - Use library [flxy](https://github.com/jcs-elpa/flxy) as matching engine.
- *`fuz-skim`* - Use library [fuz.el](https://github.com/rustify-emacs/fuz.el)'s skim algorithm.
- *`fuz-clangd`* - Use library [fuz.el](https://github.com/rustify-emacs/fuz.el)'s clangd algorithm.
- *`fuz-bin-skim`* - Use library [fuz-bin](https://github.com/jcs-elpa/fuz-bin)'s skim algorithm.
- *`fuz-bin-clangd`* - Use library [fuz-bin](https://github.com/jcs-elpa/fuz-bin)'s clangd algorithm.
- *`liquidmetal`* - Use library [liquidmetal](https://github.com/jcs-elpa/liquidmetal) similar to [Quicksilver](https://qsapp.com/) algorithm.
- *`sublime-fuzzy`* - Use library [sublime-fuzzy](https://github.com/jcs-elpa/sublime-fuzzy) as matching engine. (Sublime Text)

Or implements your sorting algorithm yourself? Assgin the function to
`company-fuzzy-sorting-function` variable like this.

```el
(setq company-fuzzy-sorting-function (lambda (candidates)
(message "%s" candidates)
candidates)) ; Don't forget to return the candidaites!
```

### ๐Ÿ” Prefix On Top

If you wish the prefix matechs on top of all other selections, customize
this variable to `t` like the line below.

```el
(setq company-fuzzy-prefix-on-top t)
```

*P.S.
If you set `company-fuzzy-sorting-backend` to `'flx` then
you probably don't need this to be on because the `flx` scoring engine
already takes care of that!*

### ๐Ÿ” For annotation

You can toggle `company-fuzzy-show-annotation` for showing annotation or not.

```el
(setq company-fuzzy-show-annotation t)
```

You can also customize annotation using the `format` variable.

* `company-fuzzy-annotation-format` => ` <%s>`

### ๐Ÿ” Excluding

You can customize variable `company-fuzzy-passthrough-backends` to exclude some
of the backends from polluting the fuzzy matching.

```el
(setq company-fuzzy-passthrough-backends '(company-capf))
```

*P.S. This is designed to be used with semantic backends, like [lsp-mode][]
(uses [company-capf][company]), or [eglot][], etc.*

## ๐Ÿ’ฌ Details

Since [company][] grants most control to users, every company backend developer
has a different method of implementing the company backend. It is hard to manage
all backends to one by various rules.

### ๐Ÿ” Reset the selection

If you are using an intelligent fuzzy matching algorithm, it is more intuitive
to reset the selected candidate to the first candidate.

```elisp
(setq company-fuzzy-reset-selection t)
```

## Recommended Settings

The company has designed something oddly to achieve this. The plugin operates
smoothly, and I suggest configuring these company variables.

```el
(use-package company
:init
(setq company-require-match nil ; Don't require match, so you can still move your cursor as expected.
company-tooltip-align-annotations t ; Align annotation to the right side.
company-eclim-auto-save nil ; Stop eclim auto save.
company-dabbrev-downcase nil) ; No downcase when completion.
:config
;; Enable downcase only when completing the completion.
(defun jcs--company-complete-selection--advice-around (fn)
"Advice execute around `company-complete-selection' command."
(let ((company-dabbrev-downcase t))
(call-interactively fn)))
(advice-add 'company-complete-selection :around #'jcs--company-complete-selection--advice-around))
```

*P.S.
For the full configuration, you can check out my configuration
[here](https://github.com/jcs-emacs/jcs-emacs/blob/21a886dea0da6ff6459f4e853f1637bd3681a4ae/lisp/jcs-plugin.el#L160-L166).*

## โ“ FAQ

#### ๐Ÿ’ซ Why is `company-fuzzy` not working?

Try to log out the `company-backends` and make sure `company-fuzzy-all-other-backends`
is the only backend in your list. If it's not, enable `company-fuzzy-mode` to swap
out all backends and hand it over to `company-fuzzy` to manage it.

```el
(message "%s" company-backends) ; '(company-fuzzy-all-other-backends)
(message "%s" company-fuzzy--backends) ; .. backends has been handed over to `company-fuzzy`
```

#### ๐Ÿ’ซ When should I call `company-fuzzy-mode`?

You should call `company-fuzzy-mode` after you have done configured
variable `company-backends`.

```el
(setq company-backends '(company-capf company-yasnippets) ; configure backends

.. (other configuration)

(company-fuzzy-mode 1) ; enable fuzzy matching at the very last
```

#### ๐Ÿ’ซ What if I want to add backends to specific `major-mode`?

You can add any backends as long as you call `company-fuzzy-mode` at the very end
of your mode hook. You can log out variable `company-fuzzy--backends` and see what
backends are currently handled by `company-fuzzy-mode`!

Or, you can hack through by configuring variable `company-fuzzye--backends` directly
but this is not recommended since after you disable `company-fuzzy-mode` it will
not be restored back to `company-backends`. Unless you change it with variable
`company-fuzzy--recorded-backends` simutamiously so it can be restored back to
your `company-backends`' true form.

โ— **UPDATE:** You can now use the following functions to accomplish these tasks
in a much elegant way:

```elisp
(company-fuzzy-backend-add 'company-capf)
(company-fuzzy-backend-remove 'company-yasnippets)
```

#### ๐Ÿ’ซ Why do some candidates aren't showing up?

This can cause by various reasons. The common causes are:

##### ๐Ÿ”Ž 1. Cause by Semantic backend rules

`company-fuzzy` respects backends' rule. Meaning the candidates can be restricted
by the backend you are using. For example,

```el
(defvar my-variable) ; You declare a variable

(my-vari.. ) ; but you are trying to use the variable as a function
```

The `my-variable` would not show up since the backend thinks it should be a
function and not a variable.

##### ๐Ÿ”Ž 2. Cause by `completion-styles`

Candidates are first filtered by Emacs built-on completion engine. Try tweaking
the variable `completion-styles` with other possible options.

```elisp
(setq completion-styles '(partial-completion))
```

Or hook up with the company's hooks:

```elisp
(add-hook 'company-completion-started-hook
(lambda ()
(setq completion-styles '(partial-completion))))

(add-hook 'company-after-completion-hook
(lambda ()
;; Revert `completion-styles' to original values
(setq completion-styles ..)))
```

See [Completion Alternatives](https://www.gnu.org/software/emacs/manual/html_node/emacs/Completion-Styles.html)
for more information.

##### ๐Ÿ”Ž 3. Scores lower than 0

Another cause would be the candidate has been eliminated by the scoring engine
(it scores lower than 0); hence it would not be shown.

Best way to debug this, is to feed `query` and `candidate` to the scoring
functions. The following example uses `flx`:

```elisp
(flx-score "window-system" "win-sys") ; return score and it's match data
```

Another example using the `liquidmetal`:

```elisp
(liquidmetal-score "window-system" "win-sys") ; return score
```

#### ๐Ÿ’ซ Why are some variables not respected with special symbols? (`@`, `:`, etc)

`company-fuzzy` detects prefix depends on the [Syntax Table](https://www.emacswiki.org/emacs/EmacsSyntaxTable)
. And those special symbols normally doesn't get treated as a whole prefix,
hence you should see the completion get inserted incorrectly,

```markdown

:lin

::link:
```

How to solve this? You should configure your syntax table by doing something
similar to

```elisp
(add-hook 'markdown-mode-hook (lambda () (modify-syntax-entry ?: "w"))
```

See related issue [#22](https://github.com/jcs-elpa/company-fuzzy/issues/22)
(`ruby-mode` with `@` symbol).

## ๐Ÿ› ๏ธ Contribute

[![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](http://makeapullrequest.com)
[![Elisp styleguide](https://img.shields.io/badge/elisp-style%20guide-purple)](https://github.com/bbatsov/emacs-lisp-style-guide)
[![Donate on paypal](https://img.shields.io/badge/paypal-donate-1?logo=paypal&color=blue)](https://www.paypal.me/jcs090218)
[![Become a patron](https://img.shields.io/badge/patreon-become%20a%20patron-orange.svg?logo=patreon)](https://www.patreon.com/jcs090218)

If you would like to contribute to this project, you may either
clone or make pull requests to this repository. Or you can
clone the project and establish your branch of this tool.
Any methods are welcome!

### ๐Ÿ”ฌ Development

To run the test locally, you will need the following tools:

- [Eask](https://emacs-eask.github.io/)
- [Make](https://www.gnu.org/software/make/) (optional)

Install all dependencies and development dependencies:

```sh
eask install-deps --dev
```

To test the package's installation:

```sh
eask package
eask install
```

To test compilation:

```sh
eask compile
```

**๐Ÿชง The following steps are optional, but we recommend you follow these lint results!**

The built-in `checkdoc` linter:

```sh
eask lint checkdoc
```

The standard `package` linter:

```sh
eask lint package
```

*๐Ÿ“ P.S. For more information, find the Eask manual at https://emacs-eask.github.io/.*

## โšœ๏ธ License

This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program. If not, see .

See [`LICENSE`](./LICENSE.txt) for details.

[company]: https://github.com/company-mode/company-mode
[lsp-mode]: https://github.com/emacs-lsp/lsp-mode
[eglot]: https://github.com/joaotavora/eglot