;;; grizzl.el --- Fast fuzzy search index for Emacs. -*- lexical-binding: t -*- ;; Copyright © 2013-2014 Chris Corbyn ;; Copyright © 2015 Bozhidar Batsov ;; ;; Author: Chris Corbyn <chris@w3style.co.uk> ;; Maintainer: Bozhidar Batsov <bozhidar@batsov.com> ;; URL: https://github.com/grizzl/grizzl ;; Package-Version: 20160130.2351 ;; Version: 0.1.2 ;; Keywords: convenience, usability ;; Package-Requires: ((cl-lib "0.5") (emacs "24.3")) ;; 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 <http://www.gnu.org/licenses/>. ;; This file is not part of GNU Emacs. ;;; Commentary: ;; Grizzl provides a fuzzy completion framework for general purpose ;; use in Emacs Lisp projects. ;; ;; grizzl provides the underlying data structures and sesrch ;; algorithm without any UI attachment. At the core, a fuzzy search ;; index is created from a list of strings, using `grizzl-make-index'. ;; A fuzzy search term is then used to get a result from this index ;; with `grizzl-search'. Because grizzl considers the usage of a ;; fuzzy search index to operate in real-time as a user enters a ;; search term in the minibuffer, the framework optimizes for this use ;; case. Any result can be passed back into `grizzl-search' as a hint ;; to continue searching. The search algorithm is able to understand ;; insertions and deletions and therefore minimizes the work it needs ;; to do in this case. The intended use here is to collect a result ;; on each key press and feed that result into the search for the next ;; key press. Once a search is complete, the matched strings are then ;; read, using `grizzl-result-strings'. The results are ordered on the ;; a combination of the Levenshtein Distance and a character-proximity ;; scoring calculation. This means shorter strings are favoured, but ;; adjacent letters are more heavily favoured. ;; ;; It is assumed that the index will be re-used across multiple ;; searches on larger sets of data. ;; ;; Call `grizzl-completing-read' with an index returned by ;; `grizzl-make-index': ;; ;; (defvar *index* (grizzl-make-index '("one" "two" "three"))) ;; (grizzl-completing-read "Number: " *index*) ;; ;; When the user hits ENTER, either one of the strings is returned on ;; success, or nil of nothing matched. ;; ;; The arrow keys can be used to navigate within the results. ;;; Code: (eval-when-compile (require 'cl-lib)) ;;; --- Public Functions ;;;###autoload (defun grizzl-make-index (strings &rest options) "Makes an index from the list STRINGS for use with `grizzl-search'. If :PROGRESS-FN is given as a keyword argument, it is called repeatedly with integers N and TOTAL. If :CASE-SENSITIVE is specified as a non-nil keyword argument, the index will be created case-sensitive, otherwise it will be case-insensitive." (let ((lookup-table (make-hash-table)) (total-strs (length strings)) (case-sensitive (plist-get options :case-sensitive)) (progress-fn (plist-get options :progress-fn)) (string-data (vconcat (mapcar (lambda (s) (cons s (length s))) strings)))) (cl-reduce (lambda (list-offset str) (grizzl-index-insert str list-offset lookup-table :case-sensitive case-sensitive) (when progress-fn (funcall progress-fn (1+ list-offset) total-strs)) (1+ list-offset)) strings :initial-value 0) (maphash (lambda (_char str-map) (maphash (lambda (list-offset locations) (puthash list-offset (reverse locations) str-map)) str-map)) lookup-table) `((case-sensitive . ,case-sensitive) (lookup-table . ,lookup-table) (string-data . ,string-data)))) ;;;###autoload (defun grizzl-search (term index &optional old-result) "Fuzzy searches for TERM in INDEX prepared with `grizzl-make-index'. OLD-RESULT may be specified as an existing search result to increment from. The result can be read with `grizzl-result-strings'." (let* ((cased-term (if (grizzl-index-case-sensitive-p index) term (downcase term))) (result (grizzl-rewind-result cased-term index old-result)) (matches (copy-hash-table (grizzl-result-matches result))) (from-pos (length (grizzl-result-term result))) (remainder (substring cased-term from-pos)) (lookup-table (grizzl-lookup-table index))) (cl-reduce (lambda (acc-res ch) (let ((sub-table (gethash ch lookup-table))) (if (not sub-table) (clrhash matches) (grizzl-search-increment sub-table matches)) (grizzl-cons-result cased-term matches acc-res))) remainder :initial-value result))) ;;;###autoload (defun grizzl-result-count (result) "Returns the number of matches present in RESULT." (hash-table-count (grizzl-result-matches result))) ;;;###autoload (defun grizzl-result-strings (result index &rest options) "Returns the ordered list of matched strings in RESULT, using INDEX. If the :START option is specified, results are read from the given offset. If the :END option is specified, up to :END results are returned." (let* ((matches (grizzl-result-matches result)) (strings (grizzl-index-strings index)) (loaded '())) (maphash (lambda (string-offset _char-offset) (push string-offset loaded)) matches) (let* ((ordered (sort loaded (lambda (a b) (< (cadr (gethash a matches)) (cadr (gethash b matches)))))) (start (or (plist-get options :start) 0)) (end (min (plist-get options :end) (length ordered))) (best (if (or start end) (cl-delete-if-not 'identity (cl-subseq ordered start end)) ordered))) (mapcar (lambda (n) (car (elt strings n))) best)))) ;;; --- Private Functions (defun grizzl-cons-result (term matches results) "Build a new result for TERM and hash-table MATCHES consed with RESULTS." (cons (cons term matches) results)) (defun grizzl-rewind-result (term index result) "Adjusts RESULT according to TERM, ready for a new search." (if result (let* ((old-term (grizzl-result-term result)) (new-len (length term)) (old-len (length old-term))) (if (and (>= new-len old-len) (string-equal old-term (substring term 0 old-len))) result (grizzl-rewind-result term index (cdr result)))) (grizzl-cons-result "" (grizzl-base-matches index) nil))) (defun grizzl-base-matches (index) "Returns the full set of matches in INDEX, with an out-of-bound offset." (let ((matches (make-hash-table))) (cl-reduce (lambda (n s-len) (puthash n (list -1 0 (cdr s-len)) matches) (1+ n)) (grizzl-index-strings index) :initial-value 0) matches)) (defun grizzl-result-term (result) "Returns the search term used to find the matches in RESULT." (car (car result))) (defun grizzl-result-matches (result) "Returns the internal hash used to track the matches in RESULT." (cdar result)) (defun grizzl-index-insert (string list-offset index &rest options) "Inserts STRING at LIST-OFFSET into INDEX." (let ((case-sensitive (plist-get options :case-sensitive))) (cl-reduce (lambda (char-offset cs-char) (let* ((char (if case-sensitive cs-char (downcase cs-char))) (str-map (or (gethash char index) (puthash char (make-hash-table) index))) (offsets (gethash list-offset str-map))) (puthash list-offset (cons char-offset offsets) str-map) (1+ char-offset))) string :initial-value 0))) (defun grizzl-lookup-table (index) "Returns the lookup table portion of INDEX." (cdr (assoc 'lookup-table index))) (defun grizzl-index-strings (index) "Returns the vector of strings stored in INDEX." (cdr (assoc 'string-data index))) (defun grizzl-index-case-sensitive-p (index) "Predicate to test of INDEX is case-sensitive." (cdr (assoc 'case-sensitive index))) (defun grizzl-search-increment (sub-table result) "Use the search lookup table to filter already-accumulated results." (cl-flet ((next-offset (key current sub-table) (cl-find-if (lambda (v) (> v current)) (gethash key sub-table)))) (maphash (lambda (k v) (let* ((oldpos (car v)) (oldrank (cadr v)) (len (cl-caddr v)) (newpos (next-offset k oldpos sub-table))) (if newpos (puthash k (list newpos (grizzl-inc-rank oldrank oldpos newpos len) len) result) (remhash k result)))) result))) (defun grizzl-inc-rank (oldrank oldpos newpos len) "Increment the current match distance as a new char is matched." (let ((distance (if (< oldpos 0) 1 (- newpos oldpos)))) (+ oldrank (* len (* distance distance))))) ;;; --- Configuration Variables (defvar *grizzl-read-max-results* 10 "The maximum number of results to show in `grizzl-completing-read'.") ;;; --- Runtime Processing Variables (defvar *grizzl-current-result* nil "The search result in `grizzl-completing-read'.") (defvar *grizzl-current-selection* 0 "The selected offset in `grizzl-completing-read'.") (defface grizzl-selection-face `((((class color) (background light)) (:foreground "red")) (((class color) (background dark)) (:foreground "red")) (t (:foreground "red"))) "Face for selected result." :group 'grizzl-mode) ;;; --- Minor Mode Definition (defvar *grizzl-keymap* (make-sparse-keymap) "Internal keymap used by the minor-mode in `grizzl-completing-read'.") (define-key *grizzl-keymap* (kbd "<up>") 'grizzl-set-selection+1) (define-key *grizzl-keymap* (kbd "C-p") 'grizzl-set-selection+1) (define-key *grizzl-keymap* (kbd "<down>") 'grizzl-set-selection-1) (define-key *grizzl-keymap* (kbd "C-n") 'grizzl-set-selection-1) (define-minor-mode grizzl-mode "Toggle the internal mode used by `grizzl-completing-read'." nil " Grizzl" *grizzl-keymap*) ;;; --- Public Functions ;;;###autoload (defun grizzl-completing-read (prompt index) "Performs a completing-read in the minibuffer using INDEX to fuzzy search. Each key pressed in the minibuffer filters down the list of matches." (minibuffer-with-setup-hook (lambda () (setq *grizzl-current-result* nil) (setq *grizzl-current-selection* 0) (grizzl-mode 1) (let* ((hookfun (lambda () (setq *grizzl-current-result* (grizzl-search (minibuffer-contents) index *grizzl-current-result*)) (grizzl-display-result index prompt))) (exitfun (lambda () (grizzl-mode -1) (remove-hook 'post-command-hook hookfun t)))) (add-hook 'minibuffer-exit-hook exitfun nil t) (add-hook 'post-command-hook hookfun nil t))) (let ((read-value (read-from-minibuffer ">>> "))) (or (grizzl-selected-result index) read-value)))) ;;;###autoload (defun grizzl-selected-result (index) "Get the selected string from INDEX in a `grizzl-completing-read'." (elt (grizzl-result-strings *grizzl-current-result* index :start 0 :end *grizzl-read-max-results*) (grizzl-current-selection))) ;;;###autoload (defun grizzl-set-selection+1 () "Move the selection up one row in `grizzl-completing-read'." (interactive) (grizzl-move-selection 1)) ;;;###autoload (defun grizzl-set-selection-1 () "Move the selection down one row in `grizzl-completing-read'." (interactive) (grizzl-move-selection -1)) ;;; --- Private Functions (defun grizzl-move-selection (delta) "Move the selection by DELTA rows in `grizzl-completing-read'." (setq *grizzl-current-selection* (+ (grizzl-current-selection) delta)) (when (not (= (grizzl-current-selection) *grizzl-current-selection*)) (beep))) (defun grizzl-display-result (index prompt) "Renders a series of overlays to list the matches in the result." (let* ((matches (grizzl-result-strings *grizzl-current-result* index :start 0 :end *grizzl-read-max-results*))) (delete-all-overlays) (overlay-put (make-overlay (point-min) (point-min)) 'before-string (format "%s\n%s\n" (mapconcat 'identity (grizzl-map-format-matches matches) "\n") (grizzl-format-prompt-line prompt))) (set-window-text-height nil (max 3 (+ 2 (length matches)))))) (defun grizzl-map-format-matches (matches) "Convert the set of string MATCHES into propertized text objects." (if (= 0 (length matches)) (list (propertize "-- NO MATCH --" 'face 'outline-3)) (cdr (cl-reduce (lambda (acc str) (let* ((idx (car acc)) (lst (cdr acc)) (sel (= idx (grizzl-current-selection)))) (cons (1+ idx) (cons (grizzl-format-match str sel) lst)))) matches :initial-value '(0))))) (defun grizzl-format-match (match-str selected) "Default match string formatter in `grizzl-completing-read'. MATCH-STR is the string in the selection list and SELECTED is non-nil if this is the current selection." (let ((margin (if selected "> " " ")) (face (if selected 'grizzl-selection-face 'default))) (propertize (format "%s%s" margin match-str) 'face face))) (defun grizzl-format-prompt-line (prompt) "Returns a string to render a full-width prompt in `grizzl-completing-read'." (let* ((count (grizzl-result-count *grizzl-current-result*)) (match-info (format " (%d candidate%s) ---- *-" count (if (= count 1) "" "s")))) (concat (propertize (format "-*%s *-" prompt) 'face 'modeline-inactive) (propertize " " 'face 'modeline-inactive 'display `(space :align-to (- right ,(1+ (length match-info))))) (propertize match-info 'face 'modeline-inactive)))) (defun grizzl-current-selection () "Get the currently selected index in `grizzl-completing-read'." (let ((max-selection (min (1- *grizzl-read-max-results*) (1- (grizzl-result-count *grizzl-current-result*))))) (max 0 (min max-selection *grizzl-current-selection*)))) (provide 'grizzl) ;;; grizzl.el ends here