forked from kostafey/ejc-sql
-
Notifications
You must be signed in to change notification settings - Fork 0
/
ejc-eldoc.el
143 lines (128 loc) · 5.2 KB
/
ejc-eldoc.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
;;; ejc-eldoc.el -- ejc-sql eldoc support (the part of ejc-sql).
;;; Copyright (C) 2019 - Kostafey <[email protected]>
;;; 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 2, 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, write to the Free Software Foundation,
;;; Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
;;; Code:
(require 'dash)
(require 'eldoc)
(require 'ejc-format)
(defun ejc-replace-property-mark (text fmt face)
(while (string-match fmt text)
(let ((beg (match-beginning 0))
(end (match-end 0)))
(setq text (concat (substring text 0 beg)
(substring text (1+ beg))))
(add-face-text-property beg (1- end) face t text)))
text)
(defun ejc-propertize (text)
(-> text
(ejc-replace-property-mark "\\@\\(\\w+_?\\)+"
'font-lock-function-name-face)
(ejc-replace-property-mark "\\%\\(\\w+_?\\)+"
'font-lock-keyword-face)
(ejc-replace-property-mark "\\#\\(\\w+_?\\)+"
'eldoc-highlight-function-argument)))
(defconst ejc-sql-expressions
(list
"SELECT"
"%SELECT #field... %FROM table [%WHERE predicate]"
"FROM"
"%SELECT field... %FROM #table [%WHERE predicate]"
"WHERE"
"%WHERE #predicate [%OR predicate] [%AND predicate]"))
(defun ejc-get-procedure-before-point ()
"Return stored procedure name before the point."
(interactive)
(save-excursion
(goto-char (nth 1 (syntax-ppss)))
(thing-at-point 'symbol)))
(defun ejc-get-package-before-point ()
"Return package name of stored procedure before the point."
(interactive)
(save-excursion
(goto-char (nth 1 (syntax-ppss)))
(beginning-of-thing 'symbol)
(when (equal (string (char-before)) ".")
(left-char)
(thing-at-point 'symbol))))
(defun ejc-get-parameter-index ()
"Return parameter number around the point."
(interactive)
(let ((index 0)
(ch (string (preceding-char))))
(save-excursion
(while (nth 2 (syntax-ppss))
(let ((pss (nth 2 (syntax-ppss))))
(when (member ch (list " " "\t" "\n" ","))
(setq index (1+ index))
(setq ch nil))
(goto-char pss)
(setq index (1+ index)))))
(max (1- index) 0)))
(defun ejc-get-word-before-point ()
"Return SQL word before the point."
(interactive)
(save-excursion
(if (not (member (string (preceding-char)) (list " " "\t" "\n")))
(forward-same-syntax -1))
(goto-char (nth 2 (syntax-ppss)))
(thing-at-point 'symbol)))
(defun ejc-eldoc-function ()
"Returns a doc string appropriate for the current context, or nil."
(if-let ((stored-procedure (if (ejc-buffer-connected-p)
(condition-case nil
(ejc-get-procedure-before-point)
(error nil)))))
(let ((type (ejc-get-entity-type ejc-db stored-procedure))
(package (ejc-get-package-before-point)))
(if (or (eq type :procedure)
(eq type :function))
(let ((params (car (ejc-get-parameters ejc-db
package
stored-procedure
t)))
(p-index (ejc-get-parameter-index)))
(ejc-propertize
(format "@%s: (%s)"
stored-procedure
(string-join
(-map
(lambda (p) (if (eql (cdr p) p-index)
(ejc-split-and-join
(lambda (s) (concat "#" s ))
" ")
(car p)))
(-zip params
(number-sequence 0 (1- (length params)))))
", "))))))
(if-let ((sql-word (condition-case nil
(ejc-get-word-before-point)
(error nil))))
(if-let ((sql-expression (lax-plist-get
ejc-sql-expressions
sql-word)))
(ejc-propertize sql-expression)))))
;;;###autoload
(defun ejc-eldoc-setup ()
"Set up eldoc function and enable eldoc-mode."
(interactive)
(cond
((boundp 'eldoc-documentation-strategy)
(setq-local eldoc-documentation-strategy #'ejc-eldoc-function))
((boundp 'eldoc-documentation-functions)
(add-hook 'eldoc-documentation-functions #'ejc-eldoc-function nil t))
(t (setq-local eldoc-documentation-function #'ejc-eldoc-function)))
(eldoc-mode +1))
(provide 'ejc-eldoc)
;;; ejc-eldoc.el ends here