X-Git-Url: https://code.delx.au/gnu-emacs-elpa/blobdiff_plain/c72773c7f45158568754c7d639fcc25f268b7c9a..ada2ede3ec75235ef0a6721e526b2fc94284093e:/company-gtags.el diff --git a/company-gtags.el b/company-gtags.el index 196050dfc..50507838e 100644 --- a/company-gtags.el +++ b/company-gtags.el @@ -1,79 +1,116 @@ -;;; company-gtags.el --- a company-mode completion back-end for GNU Global -;; -;; Copyright (C) 2009 Nikolaj Schumacher -;; -;; This file is part of company 0.4.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 2 -;; of the License, or (at your option) any later version. -;; -;; This program is distributed in the hope that it will be useful, +;;; company-gtags.el --- company-mode completion backend for GNU Global + +;; Copyright (C) 2009-2011, 2014 Free Software Foundation, Inc. + +;; Author: Nikolaj Schumacher + +;; This file is part of GNU Emacs. + +;; GNU Emacs 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. + +;; GNU Emacs 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 . +;; along with GNU Emacs. If not, see . + + +;;; Commentary: +;; + +;;; Code: (require 'company) -(eval-when-compile (require 'cl)) +(require 'company-template) +(require 'cl-lib) + +(defgroup company-gtags nil + "Completion backend for GNU Global." + :group 'company) (defcustom company-gtags-executable (executable-find "global") - "*Location of GNU global executable" - :type 'string - :group 'company) + "Location of GNU global executable." + :type 'string) (define-obsolete-variable-alias 'company-gtags-gnu-global-program-name - 'company-gtags-executable) + 'company-gtags-executable "earlier") + +(defcustom company-gtags-insert-arguments t + "When non-nil, insert function arguments as a template after completion." + :type 'boolean + :package-version '(company . "0.8.1")) -(defvar company-gtags--tags-available-p 'unknown) -(make-variable-buffer-local 'company-gtags--tags-available-p) +(defvar-local company-gtags--tags-available-p 'unknown) -(defvar company-gtags-modes '(c-mode c++-mode jde-mode java-mode php-mode)) +(defcustom company-gtags-modes '(prog-mode jde-mode) + "Modes that use `company-gtags'. +In all these modes (and their derivatives) `company-gtags' will perform +completion." + :type '(repeat (symbol :tag "Major mode")) + :package-version '(company . "0.8.4")) (defun company-gtags--tags-available-p () (if (eq company-gtags--tags-available-p 'unknown) (setq company-gtags--tags-available-p - (company-locate-dominating-file buffer-file-name "GTAGS")) + (locate-dominating-file buffer-file-name "GTAGS")) company-gtags--tags-available-p)) -(defun company-gtags-fetch-tags (prefix) +(defun company-gtags--fetch-tags (prefix) (with-temp-buffer (let (tags) (when (= 0 (call-process company-gtags-executable nil - (list (current-buffer) nil) nil "-c" prefix)) + (list (current-buffer) nil) nil "-xGq" (concat "^" prefix))) (goto-char (point-min)) - (split-string (buffer-string) "\n" t))))) + (cl-loop while + (re-search-forward (concat + "^" + "\\([^ ]*\\)" ;; completion + "[ \t]+\\([[:digit:]]+\\)" ;; linum + "[ \t]+\\([^ \t]+\\)" ;; file + "[ \t]+\\(.*\\)" ;; definition + "$" + ) nil t) + collect + (propertize (match-string 1) + 'meta (match-string 4) + 'location (cons (expand-file-name (match-string 3)) + (string-to-number (match-string 2))) + )))))) -(defun company-gtags-location (tag) - (with-temp-buffer - (when (= 0 (call-process company-gtags-executable nil - (list (current-buffer) nil) nil "-x" tag)) - (goto-char (point-min)) - (when (looking-at (concat (regexp-quote tag) - "[ \t]+\\([[:digit:]]+\\)" - "[ \t]+\\([^ \t]+\\)")) - (cons (expand-file-name (match-string 2)) - (string-to-number (match-string 1))))))) +(defun company-gtags--annotation (arg) + (let ((meta (get-text-property 0 'meta arg))) + (when (string-match (concat arg "\\((.*)\\).*") meta) + (match-string 1 meta)))) ;;;###autoload (defun company-gtags (command &optional arg &rest ignored) - "A `company-mode' completion back-end for GNU Global." + "`company-mode' completion backend for GNU Global." (interactive (list 'interactive)) - (case command - ('interactive (company-begin-backend 'company-gtags)) - ('prefix (and company-gtags-executable - (memq major-mode company-gtags-modes) - (not (company-in-string-or-comment)) - (company-gtags--tags-available-p) - (or (company-grab-symbol) 'stop))) - ('candidates (company-gtags-fetch-tags arg)) - ('sorted t) - ('location (company-gtags-location arg)))) + (cl-case command + (interactive (company-begin-backend 'company-gtags)) + (prefix (and company-gtags-executable + buffer-file-name + (apply #'derived-mode-p company-gtags-modes) + (not (company-in-string-or-comment)) + (company-gtags--tags-available-p) + (or (company-grab-symbol) 'stop))) + (candidates (company-gtags--fetch-tags arg)) + (sorted t) + (duplicates t) + (annotation (company-gtags--annotation arg)) + (meta (get-text-property 0 'meta arg)) + (location (get-text-property 0 'location arg)) + (post-completion (let ((anno (company-gtags--annotation arg))) + (when (and company-gtags-insert-arguments anno) + (insert anno) + (company-template-c-like-templatify anno)))))) (provide 'company-gtags) ;;; company-gtags.el ends here