-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathcompany-dict.el
153 lines (130 loc) · 5.49 KB
/
company-dict.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
;;; company-dict.el --- A backend that emulates ac-source-dictionary
;;
;; Copyright (C) 2015-16 Henrik Lissner
;; Author: Henrik Lissner <http://github/hlissner>
;; Maintainer: Henrik Lissner <henrik@lissner.net>
;; Created: June 21, 2015
;; Modified: March 1, 2019
;; Version: 1.2.8
;; Keywords: company dictionary ac-source-dictionary
;; Homepage: https://github.com/hlissner/emacs-company-dict
;; Package-Requires: ((emacs "24.4") (company "0.8.12") (parent-mode "2.3"))
;;
;; This file is not part of GNU Emacs.
;;
;;; Code:
(require 'company)
(eval-when-compile
(require 'cl-lib)
(require 'subr-x))
(autoload 'parent-mode-list "parent-mode")
;; For compiler
(defvar yas-minor-mode)
(declare-function yas-expand-snippet "yasnippet")
(defgroup company-dict nil
"A backend that mimics ac-source-dictionary, with support for annotations and
documentation."
:prefix "company-dict-"
:group 'company)
(defcustom company-dict-dir (concat user-emacs-directory "dict/")
"Directory to look for dictionary files."
:group 'company-dict
:type 'directory)
(defcustom company-dict-minor-mode-list '()
"A list of minor modes to be aware of when looking up dictionaries (if they're active)."
:group 'company-dict
:type '(repeat symbol))
(defcustom company-dict-enable-fuzzy nil
"Whether to allow fuzzy searching for company-dict."
:group 'company-dict
:type 'boolean)
(define-obsolete-variable-alias 'company-dict-fuzzy 'company-dict-enable-fuzzy "v1.2.4")
(defcustom company-dict-enable-yasnippet t
"If non-nil, company-dict autocompletions will be expanded like a `yasnippet'
snippet, but only if yasnippet is loaded and `yas-minor-mode' is enabled in the
current buffer. Otherwise, company-dict will pretend this is set to nil.
Note: yasnippet is optional and not a dependency of company-dict. You'll have to
install and enable it yourself."
:group 'company-dict
:type 'boolean)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
(defvar company-dict-table (make-hash-table :test 'equal)
"A lookup hash table that maps major (or minor) modes to lists of completion candidates.")
(defun company-dict--read-file (file-path)
(when (file-exists-p file-path)
(unless (file-readable-p file-path)
(user-error "Dictionary file isn't readable! (%s)" file-path))
(decode-coding-string
(with-temp-buffer
(set-buffer-multibyte nil)
(setq buffer-file-coding-system 'utf-8)
(insert-file-contents-literally file-path)
(buffer-substring-no-properties (point-min) (point-max))) 'utf-8)))
(defun company-dict--relevant-modes ()
(append '(all) (parent-mode-list major-mode) company-dict-minor-mode-list))
(defun company-dict--relevant-dicts ()
"Merge all dicts together into one large list."
(append (gethash 'all company-dict-table)
(gethash major-mode company-dict-table)
(cl-loop for mode in company-dict-minor-mode-list
if (and (boundp mode)
(symbol-value mode))
nconc (gethash mode company-dict-table))
nil))
(defun company-dict--init (mode)
"Read dict files and populate dictionary."
(let (file contents result)
(unless (symbolp mode)
(error "Expected symbol, got %s" mode))
(unless (gethash mode company-dict-table)
(setq file (expand-file-name (symbol-name mode) company-dict-dir)
contents (company-dict--read-file file))
(when (stringp contents)
(puthash mode
(cl-loop for line in (split-string contents "\n" t)
for (label note meta) = (split-string (string-trim-right line) "\t" t)
collect (propertize label :note note :meta meta))
company-dict-table)))
result))
(defun company-dict--annotation (data)
(get-text-property 0 :note data))
(defun company-dict--meta (data)
(get-text-property 0 :meta data))
(defun company-dict--quickhelp-string (data)
(get-text-property 0 :meta data))
(defun company-dict--post-completion (data)
(when (and company-dict-enable-yasnippet
(featurep 'yasnippet)
(bound-and-true-p yas-minor-mode))
(yas-expand-snippet data (- (point) (length data)) (point))))
;;;###autoload
(defun company-dict-refresh ()
"Refresh all loaded dictionaries."
(interactive)
(let ((modes (hash-table-keys company-dict-table)))
(setq company-dict-table (make-hash-table :test 'equal))
(mapc 'company-dict--init modes)))
;;;###autoload
(defun company-dict (command &optional arg &rest ignored)
"`company-mode' backend for user-provided dictionaries. Dictionary files are lazy
loaded."
(interactive (list 'interactive))
(mapc 'company-dict--init (company-dict--relevant-modes))
(let ((dicts (company-dict--relevant-dicts)))
(cl-case command
(interactive (company-begin-backend 'company-dict))
(prefix (and dicts (company-grab-symbol)))
(candidates (cl-remove-if-not
(if company-dict-enable-fuzzy
(lambda (c) (cl-subsetp (string-to-list arg)
(string-to-list c)))
(lambda (c) (string-prefix-p arg c)))
dicts))
(annotation (company-dict--annotation arg))
(meta (company-dict--meta arg))
(quickhelp-string (company-dict--quickhelp-string arg))
(post-completion (company-dict--post-completion arg))
(sorted 't)
(no-cache 't))))
(provide 'company-dict)
;;; company-dict.el ends here