;;; helm-company.el --- Helm interface for company-mode ;; Copyright (C) 2013 Yasuyuki Oka <yasuyk@gmail.com> ;; Author: Yasuyuki Oka <yasuyk@gmail.com> ;; Version: 0.1.1 ;; Package-Version: 20160516.2258 ;; URL: https://github.com/yasuyk/helm-company ;; Package-Requires: ((helm "1.5.9") (company "0.6.13")) ;; 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: ;; Add the following to your Emacs init file: ;; ;; (autoload 'helm-company "helm-company") ;; Not necessary if using ELPA package ;; (eval-after-load 'company ;; '(progn ;; (define-key company-mode-map (kbd "C-:") 'helm-company) ;; (define-key company-active-map (kbd "C-:") 'helm-company))) ;;; Code: (require 'helm) (require 'helm-multi-match) (require 'helm-files) (require 'helm-elisp) ;; For with-helm-show-completion (require 'company) (defgroup helm-company nil "Helm interface for company-mode." :prefix "helm-company-" :group 'helm) (defcustom helm-company-candidate-number-limit 300 "Limit candidate number of `helm-company'. Set it to nil if you don't want this limit." :group 'helm-company :type '(choice (const :tag "Disabled" nil) integer)) (defvar helm-company-help-window nil) (defvar helm-company-backend nil) (defun helm-company-call-backend (&rest args) "Bridge between helm-company and company" (let ((company-backend helm-company-backend)) (apply 'company-call-backend args))) (defun helm-company-init () "Prepare helm for company." (helm-attrset 'company-candidates company-candidates) (helm-attrset 'company-common company-common) (setq helm-company-help-window nil) (if (<= (length company-candidates) 1) (helm-exit-minibuffer) (setq helm-company-backend company-backend helm-company-candidates company-candidates)) (company-abort)) (defun helm-company-action-insert (candidate) "Insert CANDIDATE." (delete-char (- (length (helm-attr 'company-common)))) (insert candidate) ;; for GC (helm-attrset 'company-candidates nil)) (defun helm-company-action-show-document (candidate) "Show the documentation of the CANDIDATE." (interactive) (let ((selection (cl-find-if (lambda (s) (string-match-p candidate s)) helm-company-candidates)) (buffer (helm-company-call-backend 'doc-buffer selection))) (when buffer (display-buffer buffer)))) (defun helm-company-show-doc-buffer (candidate) "Temporarily show the documentation buffer for the CANDIDATE." (interactive) (let* ((selection (cl-find-if (lambda (s) (string-match-p candidate s)) helm-company-candidates)) (buffer (helm-company-call-backend 'doc-buffer selection))) (when buffer (if (and helm-company-help-window (window-live-p helm-company-help-window)) (with-selected-window helm-company-help-window (helm-company-display-persistent-buffer buffer)) (setq helm-company-help-window (helm-company-display-persistent-buffer buffer)))))) (defun helm-company-find-location (candidate) "Find location of CANDIDATE." (interactive) (let* ((selection (cl-find-if (lambda (s) (string-match-p candidate s)) helm-company-candidates)) (location (save-excursion (helm-company-call-backend 'location selection))) (pos (or (cdr location) (error "No location available"))) (buffer (or (and (bufferp (car location)) (car location)) (find-file-noselect (car location) t)))) (with-selected-window (display-buffer buffer t) (save-restriction (widen) (if (bufferp (car location)) (goto-char pos) (goto-char (point-min)) (forward-line (1- pos)))) (set-window-start nil (point))))) (defun helm-company-display-document-buffer (buffer) "Temporarily show the documentation BUFFER." (with-current-buffer buffer (goto-char (point-min))) (display-buffer buffer '((display-buffer-same-window . t) (display-buffer-reuse-window . t)))) (defmacro helm-company-run-action (&rest body) `(with-helm-window (save-selected-window (with-helm-display-same-window ,@body)))) (defun helm-company-run-show-doc-buffer () "Run showing documentation action from `helm-company'." (interactive) (helm-company-run-action (helm-company-show-doc-buffer (helm-get-selection)))) (defun helm-company-run-show-location () "Run showing location action from `helm-company'." (interactive) (helm-company-run-action (helm-company-find-location (helm-get-selection)))) (defvar helm-company-map (let ((keymap (make-sparse-keymap))) (set-keymap-parent keymap helm-map) (define-key keymap (kbd "M-s") 'helm-company-run-show-location) (define-key keymap (kbd "C-s") 'helm-company-run-show-doc-buffer) (delq nil keymap)) "Keymap used in Company sources.") (defvar helm-company-actions '(("Insert" . helm-company-action-insert) ("Show documentation (If available)" . helm-company-action-show-document) ("Find location (If available)" . helm-company-find-location)) "Actions for `helm-company'.") (defcustom helm-company-fuzzy-match t "Enable fuzzy matching for Helm Company." :type 'boolean) (defvar helm-source-company (helm-build-in-buffer-source "Company" :data (lambda () (helm-company-init) (helm-attr 'company-candidates)) :fuzzy-match helm-company-fuzzy-match :keymap helm-company-map :persistent-action 'helm-company-show-doc-buffer :persistent-help "Show documentation (If available)" :action helm-company-actions) "Helm source definition for recent files in current project.") ;;;###autoload (defun helm-company () "Select `company-complete' candidates by `helm'. It is useful to narrow candidates." (interactive) (unless company-candidates (company-complete)) (when company-point (company-complete-common) (helm :sources 'helm-source-company :buffer "*helm company*" :candidate-number-limit helm-company-candidate-number-limit))) (provide 'helm-company) ;; Local Variables: ;; coding: utf-8 ;; eval: (setq byte-compile-not-obsolete-vars '(display-buffer-function)) ;; eval: (checkdoc-minor-mode 1) ;; End: ;;; helm-company.el ends here