;;; auto-complete-clang.el --- Auto Completion source for clang for GNU Emacs ;; Copyright (C) 2010 Brian Jiang ;; Author: Brian Jiang ;; Keywords: completion, convenience ;; URL: https://github.com/brianjcj/auto-complete-clang ;; Version: 0.1i ;; Package-Requires: ((auto-complete "1.3.1")) ;; 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 . ;;; Commentary: ;; ;; Auto Completion source for clang. Most of codes are taken from ;; company-clang.el and modified and enhanced for Auto Completion. ;;; Code: (provide 'auto-complete-clang) (require 'auto-complete) (defcustom ac-clang-executable (executable-find "clang") "*Location of clang executable" :group 'auto-complete :type 'file) (defcustom ac-clang-auto-save nil "*Determines whether to save the buffer when retrieving completions. Old version of clang can only complete correctly when the buffer has been saved. Now clang can parse the codes from standard input so that we can turn this option to Off. If you are still using the old clang, turn it on!" :group 'auto-complete :type '(choice (const :tag "Off" nil) (const :tag "On" t))) (defcustom ac-clang-lang-option-function nil "*function to return the lang type for option -x." :group 'auto-complete :type 'function) ;;; Extra compilation flags to pass to clang. (defcustom ac-clang-flags nil "Extra flags to pass to the Clang executable. This variable will typically contain include paths, e.g., ( \"-I~/MyProject\", \"-I.\" )." :group 'auto-complete :type '(repeat (string :tag "Argument" ""))) ;;; The prefix header to use with Clang code completion. (defvar ac-clang-prefix-header nil) ;;; Set the Clang prefix header (defun ac-clang-set-prefix-header (ph) (interactive (let ((def (car (directory-files "." t "\\([^.]h\\|[^h]\\).pch\\'" t)))) (list (read-file-name (concat "Clang prefix header(current: " ac-clang-prefix-header ") : ") (when def (file-name-directory def)) def nil (when def (file-name-nondirectory def)))))) (cond ((string-match "^[\s\t]*$" ph) (setq ac-clang-prefix-header nil)) (t (setq ac-clang-prefix-header ph)))) ;;; Set a new cflags for clang (defun ac-clang-set-cflags () "set new cflags for clang from input string" (interactive) (setq ac-clang-flags (split-string (read-string "New cflags: ")))) ;;; Set new cflags from shell command output (defun ac-clang-set-cflags-from-shell-command () "set new cflags for ac-clang from shell command output" (interactive) (setq ac-clang-flags (split-string (shell-command-to-string (read-shell-command "Shell command: " nil nil (and buffer-file-name (file-relative-name buffer-file-name))))))) (defconst ac-clang-completion-pattern "^COMPLETION: \\(%s[^\s\n:]*\\)\\(?: : \\)*\\(.*$\\)") (defconst ac-clang-error-buffer-name "*clang error*") (defun ac-clang-parse-output (prefix) (goto-char (point-min)) (let ((pattern (format ac-clang-completion-pattern (regexp-quote prefix))) lines match detailed_info (prev-match "")) (while (re-search-forward pattern nil t) (setq match (match-string-no-properties 1)) (unless (string= "Pattern" match) (setq detailed_info (match-string-no-properties 2)) (if (string= match prev-match) (progn (when detailed_info (setq match (propertize match 'ac-clang-help (concat (get-text-property 0 'ac-clang-help (car lines)) "\n" detailed_info))) (setf (car lines) match) )) (setq prev-match match) (when detailed_info (setq match (propertize match 'ac-clang-help detailed_info))) (push match lines)))) lines)) (defun ac-clang-handle-error (res args) (goto-char (point-min)) (let* ((buf (get-buffer-create ac-clang-error-buffer-name)) (cmd (concat ac-clang-executable " " (mapconcat 'identity args " "))) (pattern (format ac-clang-completion-pattern "")) (err (if (re-search-forward pattern nil t) (buffer-substring-no-properties (point-min) (1- (match-beginning 0))) ;; Warn the user more agressively if no match was found. (message "clang failed with error %d:\n%s" res cmd) (buffer-string)))) (with-current-buffer buf (let ((inhibit-read-only t)) (erase-buffer) (insert (current-time-string) (format "\nclang failed with error %d:\n" res) cmd "\n\n") (insert err) (setq buffer-read-only t) (goto-char (point-min)))))) (defun ac-clang-call-process (prefix &rest args) (let ((buf (get-buffer-create "*clang-output*")) res) (with-current-buffer buf (erase-buffer)) (setq res (if ac-clang-auto-save (apply 'call-process ac-clang-executable nil buf nil args) (apply 'call-process-region (point-min) (point-max) ac-clang-executable nil buf nil args))) (with-current-buffer buf (unless (eq 0 res) (ac-clang-handle-error res args)) ;; Still try to get any useful input. (ac-clang-parse-output prefix)))) (defsubst ac-clang-build-location (pos) (save-excursion (goto-char pos) (format "%s:%d:%d" (if ac-clang-auto-save buffer-file-name "-") (line-number-at-pos) (1+ (- (point) (line-beginning-position)))))) (defsubst ac-clang-lang-option () (or (and ac-clang-lang-option-function (funcall ac-clang-lang-option-function)) (cond ((eq major-mode 'c++-mode) "c++") ((eq major-mode 'c-mode) "c") ((eq major-mode 'objc-mode) (cond ((string= "m" (file-name-extension (buffer-file-name))) "objective-c") (t "objective-c++"))) (t "c++")))) (defsubst ac-clang-build-complete-args (pos) (append '("-cc1" "-fsyntax-only") (unless ac-clang-auto-save (list "-x" (ac-clang-lang-option))) ac-clang-flags (when (stringp ac-clang-prefix-header) (list "-include-pch" (expand-file-name ac-clang-prefix-header))) '("-code-completion-at") (list (ac-clang-build-location pos)) (list (if ac-clang-auto-save buffer-file-name "-")))) (defsubst ac-clang-clean-document (s) (when s (setq s (replace-regexp-in-string "<#\\|#>\\|\\[#" "" s)) (setq s (replace-regexp-in-string "#\\]" " " s))) s) (defun ac-clang-document (item) (if (stringp item) (let (s) (setq s (get-text-property 0 'ac-clang-help item)) (ac-clang-clean-document s))) ;; (popup-item-property item 'ac-clang-help) ) (defface ac-clang-candidate-face '((t (:background "lightgray" :foreground "navy"))) "Face for clang candidate" :group 'auto-complete) (defface ac-clang-selection-face '((t (:background "navy" :foreground "white"))) "Face for the clang selected candidate." :group 'auto-complete) (defsubst ac-in-string/comment () "Return non-nil if point is in a literal (a comment or string)." (nth 8 (syntax-ppss))) (defun ac-clang-candidate () (unless (ac-in-string/comment) (and ac-clang-auto-save (buffer-modified-p) (basic-save-buffer)) (save-restriction (widen) (apply 'ac-clang-call-process ac-prefix (ac-clang-build-complete-args (- (point) (length ac-prefix))))))) (defvar ac-template-start-point nil) (defvar ac-template-candidates (list "ok" "no" "yes:)")) (defun ac-clang-action () (interactive) ;; (ac-last-quick-help) (let ((help (ac-clang-clean-document (get-text-property 0 'ac-clang-help (cdr ac-last-completion)))) (raw-help (get-text-property 0 'ac-clang-help (cdr ac-last-completion))) (candidates (list)) ss fn args (ret-t "") ret-f) (setq ss (split-string raw-help "\n")) (dolist (s ss) (when (string-match "\\[#\\(.*\\)#\\]" s) (setq ret-t (match-string 1 s))) (setq s (replace-regexp-in-string "\\[#.*?#\\]" "" s)) (cond ((string-match "^\\([^(]*\\)\\((.*)\\)" s) (setq fn (match-string 1 s) args (match-string 2 s)) (push (propertize (ac-clang-clean-document args) 'ac-clang-help ret-t 'raw-args args) candidates) (when (string-match "\{#" args) (setq args (replace-regexp-in-string "\{#.*#\}" "" args)) (push (propertize (ac-clang-clean-document args) 'ac-clang-help ret-t 'raw-args args) candidates)) (when (string-match ", \\.\\.\\." args) (setq args (replace-regexp-in-string ", \\.\\.\\." "" args)) (push (propertize (ac-clang-clean-document args) 'ac-clang-help ret-t 'raw-args args) candidates))) ((string-match "^\\([^(]*\\)(\\*)\\((.*)\\)" ret-t) ;; check whether it is a function ptr (setq ret-f (match-string 1 ret-t) args (match-string 2 ret-t)) (push (propertize args 'ac-clang-help ret-f 'raw-args "") candidates) (when (string-match ", \\.\\.\\." args) (setq args (replace-regexp-in-string ", \\.\\.\\." "" args)) (push (propertize args 'ac-clang-help ret-f 'raw-args "") candidates))))) (cond (candidates (setq candidates (delete-dups candidates)) (setq candidates (nreverse candidates)) (setq ac-template-candidates candidates) (setq ac-template-start-point (point)) (ac-complete-template) (unless (cdr candidates) ;; unless length > 1 (message (replace-regexp-in-string "\n" " ; " help)))) (t (message (replace-regexp-in-string "\n" " ; " help)))))) (defun ac-clang-prefix () (or (ac-prefix-symbol) (let ((c (char-before))) (when (or (eq ?\. c) ;; -> (and (eq ?> c) (eq ?- (char-before (1- (point))))) ;; :: (and (eq ?: c) (eq ?: (char-before (1- (point)))))) (point))))) (ac-define-source clang '((candidates . ac-clang-candidate) (candidate-face . ac-clang-candidate-face) (selection-face . ac-clang-selection-face) (prefix . ac-clang-prefix) (requires . 0) (document . ac-clang-document) (action . ac-clang-action) (cache) (symbol . "c"))) (defun ac-clang-same-count-in-string (c1 c2 s) (let ((count 0) (cur 0) (end (length s)) c) (while (< cur end) (setq c (aref s cur)) (cond ((eq c1 c) (setq count (1+ count))) ((eq c2 c) (setq count (1- count)))) (setq cur (1+ cur))) (= count 0))) (defun ac-clang-split-args (s) (let ((sl (split-string s ", *"))) (cond ((string-match "<\\|(" s) (let ((res (list)) (pre "") subs) (while sl (setq subs (pop sl)) (unless (string= pre "") (setq subs (concat pre ", " subs)) (setq pre "")) (cond ((and (ac-clang-same-count-in-string ?\< ?\> subs) (ac-clang-same-count-in-string ?\( ?\) subs)) (push subs res)) (t (setq pre subs)))) (nreverse res))) (t sl)))) (defun ac-template-candidate () ac-template-candidates) (defun ac-template-action () (interactive) (unless (null ac-template-start-point) (let ((pos (point)) sl (snp "") (s (get-text-property 0 'raw-args (cdr ac-last-completion)))) (cond ((string= s "") ;; function ptr call (setq s (cdr ac-last-completion)) (setq s (replace-regexp-in-string "^(\\|)$" "" s)) (setq sl (ac-clang-split-args s)) (cond ((featurep 'yasnippet) (dolist (arg sl) (setq snp (concat snp ", ${" arg "}"))) (condition-case nil (yas/expand-snippet (concat "(" (substring snp 2) ")") ac-template-start-point pos) ;; 0.6.1c (error ;; try this one: (ignore-errors (yas/expand-snippet ac-template-start-point pos (concat "(" (substring snp 2) ")"))) ;; work in 0.5.7 ))) ((featurep 'snippet) (delete-region ac-template-start-point pos) (dolist (arg sl) (setq snp (concat snp ", $${" arg "}"))) (snippet-insert (concat "(" (substring snp 2) ")"))) (t (message "Dude! You are too out! Please install a yasnippet or a snippet script:)")))) (t (unless (string= s "()") (setq s (replace-regexp-in-string "{#" "" s)) (setq s (replace-regexp-in-string "#}" "" s)) (cond ((featurep 'yasnippet) (setq s (replace-regexp-in-string "<#" "${" s)) (setq s (replace-regexp-in-string "#>" "}" s)) (setq s (replace-regexp-in-string ", \\.\\.\\." "}, ${..." s)) (condition-case nil (yas/expand-snippet s ac-template-start-point pos) ;; 0.6.1c (error ;; try this one: (ignore-errors (yas/expand-snippet ac-template-start-point pos s)) ;; work in 0.5.7 ))) ((featurep 'snippet) (delete-region ac-template-start-point pos) (setq s (replace-regexp-in-string "<#" "$${" s)) (setq s (replace-regexp-in-string "#>" "}" s)) (setq s (replace-regexp-in-string ", \\.\\.\\." "}, $${..." s)) (snippet-insert s)) (t (message "Dude! You are too out! Please install a yasnippet or a snippet script:)"))))))))) (defun ac-template-prefix () ac-template-start-point) ;; this source shall only be used internally. (ac-define-source template '((candidates . ac-template-candidate) (prefix . ac-template-prefix) (requires . 0) (action . ac-template-action) (document . ac-clang-document) (cache) (symbol . "t"))) ;;; auto-complete-clang.el ends here