You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
|
|
;;; company-cmake.el --- company-mode completion backend for CMake
;; Copyright (C) 2013-2014, 2017-2018 Free Software Foundation, Inc.
;; Author: Chen Bin <chenbin DOT sh AT gmail>;; Version: 0.2
;; 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/>.
;;; Commentary:;;;; company-cmake offers completions for module names, variable names and;; commands used by CMake. And their descriptions.
;;; Code:
(require 'company)(require 'cl-lib)
(defgroup company-cmake nil "Completion backend for CMake." :group 'company)
(defcustom company-cmake-executable (executable-find "cmake") "Location of cmake executable." :type 'file)
(defvar company-cmake-executable-arguments '("--help-command-list" "--help-module-list" "--help-variable-list") "The arguments we pass to cmake, separately.
They affect which types of symbols we get completion candidates for.")
(defvar company-cmake--completion-pattern "^\\(%s[a-zA-Z0-9_<>]%s\\)$" "Regexp to match the candidates.")
(defvar company-cmake-modes '(cmake-mode) "Major modes in which cmake may complete.")
(defvar company-cmake--candidates-cache nil "Cache for the raw candidates.")
(defvar company-cmake--meta-command-cache nil "Cache for command arguments to retrieve descriptions for the candidates.")
(defun company-cmake--replace-tags (rlt) (setq rlt (replace-regexp-in-string "\\(.*?\\(IS_GNU\\)?\\)<LANG>\\(.*\\)" (lambda (_match) (mapconcat 'identity (if (match-beginning 2) '("\\1CXX\\3" "\\1C\\3" "\\1G77\\3") '("\\1CXX\\3" "\\1C\\3" "\\1Fortran\\3")) "\n")) rlt t)) (setq rlt (replace-regexp-in-string "\\(.*\\)<CONFIG>\\(.*\\)" (mapconcat 'identity '("\\1DEBUG\\2" "\\1RELEASE\\2" "\\1RELWITHDEBINFO\\2" "\\1MINSIZEREL\\2") "\n") rlt)) rlt)
(defun company-cmake--fill-candidates-cache (arg) "Fill candidates cache if needed." (let (rlt) (unless company-cmake--candidates-cache (setq company-cmake--candidates-cache (make-hash-table :test 'equal)))
;; If hash is empty, fill it. (unless (gethash arg company-cmake--candidates-cache) (with-temp-buffer (let ((res (call-process company-cmake-executable nil t nil arg))) (unless (zerop res) (message "cmake executable exited with error=%d" res))) (setq rlt (buffer-string))) (setq rlt (company-cmake--replace-tags rlt)) (puthash arg rlt company-cmake--candidates-cache)) ))
(defun company-cmake--parse (prefix content cmd) (let ((start 0) (pattern (format company-cmake--completion-pattern (regexp-quote prefix) (if (zerop (length prefix)) "+" "*"))) (lines (split-string content "\n")) match rlt) (dolist (line lines) (when (string-match pattern line) (let ((match (match-string 1 line))) (when match (puthash match cmd company-cmake--meta-command-cache) (push match rlt))))) rlt))
(defun company-cmake--candidates (prefix) (let (results cmd-opts str)
(unless company-cmake--meta-command-cache (setq company-cmake--meta-command-cache (make-hash-table :test 'equal)))
(dolist (arg company-cmake-executable-arguments) (company-cmake--fill-candidates-cache arg) (setq cmd-opts (replace-regexp-in-string "-list$" "" arg) )
(setq str (gethash arg company-cmake--candidates-cache)) (when str (setq results (nconc results (company-cmake--parse prefix str cmd-opts))))) results))
(defun company-cmake--unexpand-candidate (candidate) (cond ((string-match "^CMAKE_\\(C\\|CXX\\|Fortran\\)\\(_.*\\)$" candidate) (setq candidate (concat "CMAKE_<LANG>" (match-string 2 candidate))))
;; C flags ((string-match "^\\(.*_\\)IS_GNU\\(C\\|CXX\\|G77\\)$" candidate) (setq candidate (concat (match-string 1 candidate) "IS_GNU<LANG>")))
;; C flags ((string-match "^\\(.*_\\)OVERRIDE_\\(C\\|CXX\\|Fortran\\)$" candidate) (setq candidate (concat (match-string 1 candidate) "OVERRIDE_<LANG>")))
((string-match "^\\(.*\\)\\(_DEBUG\\|_RELEASE\\|_RELWITHDEBINFO\\|_MINSIZEREL\\)\\(.*\\)$" candidate) (setq candidate (concat (match-string 1 candidate) "_<CONFIG>" (match-string 3 candidate))))) candidate)
(defun company-cmake--meta (candidate) (let ((cmd-opts (gethash candidate company-cmake--meta-command-cache)) result) (setq candidate (company-cmake--unexpand-candidate candidate))
;; Don't cache the documentation of every candidate (command) ;; Cache in this case will cost too much memory. (with-temp-buffer (call-process company-cmake-executable nil t nil cmd-opts candidate) ;; Go to the third line, trim it and return the result. ;; Tested with cmake 2.8.9. (goto-char (point-min)) (forward-line 2) (setq result (buffer-substring-no-properties (line-beginning-position) (line-end-position))) (setq result (replace-regexp-in-string "^[ \t\n\r]+" "" result)) result)))
(defun company-cmake--doc-buffer (candidate) (let ((cmd-opts (gethash candidate company-cmake--meta-command-cache)))
(setq candidate (company-cmake--unexpand-candidate candidate)) (with-temp-buffer (call-process company-cmake-executable nil t nil cmd-opts candidate) ;; Go to the third line, trim it and return the doc buffer. ;; Tested with cmake 2.8.9. (goto-char (point-min)) (forward-line 2) (company-doc-buffer (buffer-substring-no-properties (line-beginning-position) (point-max))))))
(defun company-cmake-prefix-dollar-brace-p () "Test if the current symbol follows ${." (save-excursion (skip-syntax-backward "w_") (and (eq (char-before (point)) ?\{) (eq (char-before (1- (point))) ?$))))
(defun company-cmake (command &optional arg &rest ignored) "`company-mode' completion backend for CMake.
CMake is a cross-platform, open-source make system."
(interactive (list 'interactive)) (cl-case command (interactive (company-begin-backend 'company-cmake)) (init (when (memq major-mode company-cmake-modes) (unless company-cmake-executable (error "Company found no cmake executable")))) (prefix (and (memq major-mode company-cmake-modes) (or (not (company-in-string-or-comment)) (company-cmake-prefix-dollar-brace-p)) (company-grab-symbol))) (candidates (company-cmake--candidates arg)) (meta (company-cmake--meta arg)) (doc-buffer (company-cmake--doc-buffer arg)) ))
(provide 'company-cmake);;; company-cmake.el ends here
|