2017-06-08 11:47:56 +02:00
|
|
|
;;; lang/javascript/config.el -*- lexical-binding: t; -*-
|
2017-01-16 23:30:37 -05:00
|
|
|
|
2018-09-08 18:37:52 -04:00
|
|
|
(after! (:any js2-mode rjsx-mode web-mode)
|
|
|
|
(set-docsets! '(js2-mode rjsx-mode) "JavaScript"
|
feature/lookup: rewrite dash docset integration
+ Uses alist variable to store config, rather than hooks
+ Added check for installed docsets in +lookup/documentation
+ Set docsets for various language modules (c-mode, c++-mode, css-mode,
scss-mode, sass-mode, web-mode, go-mode, racket-mode, emacs-lisp-mode,
js2-mode, rjsx-mode, typescript-mode, rust-mode, and php-mode)
+ Made *eww* popups for dash docsets larger
+ Renamed set-docset! => set-docsets! (set-docset! is aliased to
set-docsets!)
+ New +lookup/install-docset alias
2018-08-31 02:44:49 +02:00
|
|
|
"AngularJS" "Backbone" "BackboneJS" "Bootstrap" "D3JS" "EmberJS" "Express"
|
|
|
|
"ExtJS" "JQuery" "JQuery_Mobile" "JQuery_UI" "KnockoutJS" "Lo-Dash"
|
|
|
|
"MarionetteJS" "MomentJS" "NodeJS" "PrototypeJS" "React" "RequireJS"
|
|
|
|
"SailsJS" "UnderscoreJS" "VueJS" "ZeptoJS")
|
|
|
|
|
2018-09-03 23:56:25 +02:00
|
|
|
(set-pretty-symbols! '(js2-mode rjsx-mode web-mode)
|
2018-06-20 18:29:04 +10:00
|
|
|
;; Functional
|
|
|
|
:def "function"
|
|
|
|
:lambda "() =>"
|
|
|
|
:composition "compose"
|
|
|
|
;; Types
|
|
|
|
:null "null"
|
|
|
|
:true "true" :false "false"
|
|
|
|
;; Flow
|
|
|
|
:not "!"
|
|
|
|
:and "&&" :or "||"
|
|
|
|
:for "for"
|
|
|
|
:return "return"
|
|
|
|
;; Other
|
|
|
|
:yield "import"))
|
2018-06-16 19:32:25 +02:00
|
|
|
|
|
|
|
|
2018-04-20 02:44:04 -04:00
|
|
|
;;
|
|
|
|
;; Major modes
|
|
|
|
|
2017-02-23 00:06:12 -05:00
|
|
|
(def-package! js2-mode
|
2018-10-03 17:23:00 -05:00
|
|
|
:mode "\\.m?js\\'"
|
2017-02-03 19:23:29 -05:00
|
|
|
:interpreter "node"
|
2018-06-20 14:23:17 +02:00
|
|
|
:commands js2-line-break
|
2017-02-19 18:57:16 -05:00
|
|
|
:config
|
2017-01-16 23:30:37 -05:00
|
|
|
(setq js2-skip-preprocessor-directives t
|
2018-03-06 18:38:35 -05:00
|
|
|
js-chain-indent t
|
|
|
|
;; let flycheck handle this
|
2018-02-08 18:56:49 -05:00
|
|
|
js2-mode-show-parse-errors nil
|
2018-04-18 18:08:50 -04:00
|
|
|
js2-mode-show-strict-warnings nil
|
2018-04-19 00:50:47 -04:00
|
|
|
;; Flycheck provides these features, so disable them: conflicting with
|
|
|
|
;; the eslint settings.
|
2018-04-18 18:08:50 -04:00
|
|
|
js2-strict-trailing-comma-warning nil
|
2018-08-22 02:16:58 +02:00
|
|
|
js2-strict-missing-semi-warning nil
|
|
|
|
;; maximum fontification
|
|
|
|
js2-highlight-level 3
|
|
|
|
js2-highlight-external-variables t)
|
2017-01-16 23:30:37 -05:00
|
|
|
|
2018-06-21 15:54:36 +02:00
|
|
|
(add-hook 'js2-mode-hook #'rainbow-delimiters-mode)
|
2018-06-15 17:12:54 +02:00
|
|
|
;; Indent switch-case another step
|
2019-01-08 00:37:53 -05:00
|
|
|
(setq-hook! 'js2-mode-hook
|
|
|
|
js-switch-indent-offset js2-basic-offset
|
|
|
|
mode-name "JS2")
|
2018-06-15 17:12:54 +02:00
|
|
|
|
2018-06-15 12:30:56 +02:00
|
|
|
(set-electric! 'js2-mode :chars '(?\} ?\) ?. ?:))
|
2019-02-18 01:56:38 -05:00
|
|
|
(set-repl-handler! 'js2-mode #'+javascript/open-repl)
|
2017-06-19 00:25:49 +02:00
|
|
|
|
2019-03-05 19:16:08 -05:00
|
|
|
(after! projectile
|
2019-04-02 21:37:52 +02:00
|
|
|
(add-to-list 'projectile-project-root-files "package.json")
|
2019-03-05 19:16:08 -05:00
|
|
|
(add-to-list 'projectile-globally-ignored-directories "node_modules"))
|
2019-03-04 19:54:03 -05:00
|
|
|
|
2017-02-23 00:06:12 -05:00
|
|
|
(map! :map js2-mode-map
|
2017-02-19 18:57:16 -05:00
|
|
|
:localleader
|
2018-12-23 23:54:27 -05:00
|
|
|
"S" #'+javascript/skewer-this-buffer))
|
2018-04-18 18:17:17 -04:00
|
|
|
|
|
|
|
|
2018-04-20 02:44:04 -04:00
|
|
|
(def-package! rjsx-mode
|
|
|
|
:mode "components/.+\\.js$"
|
|
|
|
:init
|
|
|
|
(defun +javascript-jsx-file-p ()
|
2018-04-26 17:56:58 -04:00
|
|
|
"Detect React or preact imports early in the file."
|
2018-04-20 02:44:04 -04:00
|
|
|
(and buffer-file-name
|
|
|
|
(string= (file-name-extension buffer-file-name) "js")
|
2018-04-27 03:28:48 -04:00
|
|
|
(re-search-forward "\\(^\\s-*import +React\\|\\( from \\|require(\\)[\"']p?react\\)"
|
2018-04-20 02:44:04 -04:00
|
|
|
magic-mode-regexp-match-limit t)
|
|
|
|
(progn (goto-char (match-beginning 1))
|
|
|
|
(not (sp-point-in-string-or-comment)))))
|
2018-06-23 16:48:58 +02:00
|
|
|
(add-to-list 'magic-mode-alist '(+javascript-jsx-file-p . rjsx-mode))
|
2018-04-20 02:44:04 -04:00
|
|
|
:config
|
2018-06-15 12:30:56 +02:00
|
|
|
(set-electric! 'rjsx-mode :chars '(?\} ?\) ?. ?>))
|
2019-02-22 00:20:29 -05:00
|
|
|
(when (featurep! :tools flycheck)
|
2018-08-31 02:50:47 +02:00
|
|
|
(add-hook! 'rjsx-mode-hook
|
|
|
|
;; jshint doesn't know how to deal with jsx
|
|
|
|
(push 'javascript-jshint flycheck-disabled-checkers)))
|
2018-04-28 16:04:11 -04:00
|
|
|
|
|
|
|
;; `rjsx-electric-gt' relies on js2's parser to tell it when the cursor is in
|
|
|
|
;; a self-closing tag, so that it can insert a matching ending tag at point.
|
|
|
|
;; However, the parser doesn't run immediately, so a fast typist can outrun
|
2018-08-31 02:50:47 +02:00
|
|
|
;; it, causing tags to stay unclosed, so we force it to parse.
|
2018-04-28 16:04:11 -04:00
|
|
|
(defun +javascript|reparse (n)
|
2018-05-25 00:46:11 +02:00
|
|
|
;; if n != 1, rjsx-electric-gt calls rjsx-maybe-reparse itself
|
2018-05-22 01:47:40 +02:00
|
|
|
(if (= n 1) (rjsx-maybe-reparse)))
|
2018-04-28 16:04:11 -04:00
|
|
|
(advice-add #'rjsx-electric-gt :before #'+javascript|reparse))
|
2018-04-20 02:44:04 -04:00
|
|
|
|
|
|
|
|
2018-05-25 00:46:11 +02:00
|
|
|
(after! typescript-mode
|
2018-06-21 15:54:36 +02:00
|
|
|
(add-hook 'typescript-mode-hook #'rainbow-delimiters-mode)
|
2018-06-20 14:23:17 +02:00
|
|
|
(setq-hook! 'typescript-mode-hook
|
|
|
|
comment-line-break-function #'js2-line-break)
|
2018-06-15 12:30:56 +02:00
|
|
|
(set-electric! 'typescript-mode
|
2018-06-16 19:32:25 +02:00
|
|
|
:chars '(?\} ?\)) :words '("||" "&&"))
|
feature/lookup: rewrite dash docset integration
+ Uses alist variable to store config, rather than hooks
+ Added check for installed docsets in +lookup/documentation
+ Set docsets for various language modules (c-mode, c++-mode, css-mode,
scss-mode, sass-mode, web-mode, go-mode, racket-mode, emacs-lisp-mode,
js2-mode, rjsx-mode, typescript-mode, rust-mode, and php-mode)
+ Made *eww* popups for dash docsets larger
+ Renamed set-docset! => set-docsets! (set-docset! is aliased to
set-docsets!)
+ New +lookup/install-docset alias
2018-08-31 02:44:49 +02:00
|
|
|
(set-docsets! 'typescript-mode "TypeScript" "AngularTS")
|
2018-06-16 19:32:25 +02:00
|
|
|
(set-pretty-symbols! 'typescript-mode
|
|
|
|
;; Functional
|
|
|
|
:def "function"
|
|
|
|
:lambda "() =>"
|
|
|
|
:composition "compose"
|
|
|
|
;; Types
|
|
|
|
:null "null"
|
|
|
|
:true "true" :false "false"
|
|
|
|
:int "number"
|
|
|
|
:str "string"
|
|
|
|
:bool "boolean"
|
|
|
|
;; Flow
|
|
|
|
:not "!"
|
|
|
|
:and "&&" :or "||"
|
|
|
|
:for "for"
|
|
|
|
:return "return" :yield "import"))
|
2018-05-25 00:46:11 +02:00
|
|
|
|
|
|
|
|
|
|
|
;; `coffee-mode'
|
|
|
|
(setq coffee-indent-like-python-mode t)
|
feature/lookup: rewrite dash docset integration
+ Uses alist variable to store config, rather than hooks
+ Added check for installed docsets in +lookup/documentation
+ Set docsets for various language modules (c-mode, c++-mode, css-mode,
scss-mode, sass-mode, web-mode, go-mode, racket-mode, emacs-lisp-mode,
js2-mode, rjsx-mode, typescript-mode, rust-mode, and php-mode)
+ Made *eww* popups for dash docsets larger
+ Renamed set-docset! => set-docsets! (set-docset! is aliased to
set-docsets!)
+ New +lookup/install-docset alias
2018-08-31 02:44:49 +02:00
|
|
|
(after! coffee-mode
|
|
|
|
(set-docsets! 'coffee-mode "CoffeeScript"))
|
2018-04-20 02:44:04 -04:00
|
|
|
|
|
|
|
|
|
|
|
;;
|
|
|
|
;; Tools
|
|
|
|
|
2019-02-21 16:08:27 -05:00
|
|
|
(when (featurep! +lsp)
|
2019-03-02 01:33:54 -05:00
|
|
|
(add-hook! (js2-mode rjsx-mode typescript-mode) #'lsp!))
|
2019-02-21 16:08:27 -05:00
|
|
|
|
|
|
|
|
2018-04-18 18:04:49 -04:00
|
|
|
(def-package! tide
|
2019-02-21 16:08:27 -05:00
|
|
|
:unless (featurep! +lsp)
|
2018-06-15 17:15:25 +02:00
|
|
|
:defer t
|
2018-04-18 18:17:17 -04:00
|
|
|
:init
|
2018-06-15 17:15:25 +02:00
|
|
|
;; Don't let hard errors stop the user from opening js files.
|
|
|
|
(defun +javascript|init-tide ()
|
|
|
|
"Enable `tide-mode' if node is available."
|
2019-01-05 17:49:04 -05:00
|
|
|
(cond ((not buffer-file-name)
|
|
|
|
(add-hook 'after-save-hook #'+javascript|init-tide nil t))
|
|
|
|
((executable-find "node")
|
|
|
|
(tide-setup))
|
|
|
|
((message "Couldn't find `node', aborting tide server"))))
|
2018-06-15 17:15:25 +02:00
|
|
|
(add-hook! (js2-mode typescript-mode) #'+javascript|init-tide)
|
|
|
|
|
2018-04-18 18:17:17 -04:00
|
|
|
(defun +javascript|init-tide-in-web-mode ()
|
2018-05-25 00:46:11 +02:00
|
|
|
"Enable `tide-mode' if in a *.tsx file."
|
2018-04-18 18:17:17 -04:00
|
|
|
(when (string= (file-name-extension (or buffer-file-name "")) "tsx")
|
|
|
|
(tide-setup)))
|
|
|
|
(add-hook 'web-mode-hook #'+javascript|init-tide-in-web-mode)
|
2018-04-18 18:04:49 -04:00
|
|
|
:config
|
2018-04-29 22:52:32 -04:00
|
|
|
(setq tide-completion-detailed t
|
|
|
|
tide-always-show-documentation t)
|
2018-04-22 23:56:29 -04:00
|
|
|
;; code completion
|
|
|
|
(after! company
|
|
|
|
;; tide affects the global `company-backends', undo this so doom can handle
|
|
|
|
;; it buffer-locally
|
|
|
|
(setq-default company-backends (delq 'company-tide (default-value 'company-backends))))
|
2018-06-15 02:58:12 +02:00
|
|
|
(set-company-backend! 'tide-mode 'company-tide)
|
2018-04-22 23:56:29 -04:00
|
|
|
;; navigation
|
2019-01-08 00:33:38 -05:00
|
|
|
(set-lookup-handlers! 'tide-mode :async t
|
2018-04-18 18:04:49 -04:00
|
|
|
:definition #'tide-jump-to-definition
|
2019-03-18 01:08:59 -04:00
|
|
|
:references #'tide-references)
|
2018-04-19 00:51:51 -04:00
|
|
|
;; resolve to `doom-project-root' if `tide-project-root' fails
|
|
|
|
(advice-add #'tide-project-root :override #'+javascript*tide-project-root)
|
|
|
|
;; cleanup tsserver when no tide buffers are left
|
|
|
|
(add-hook! 'tide-mode-hook
|
|
|
|
(add-hook 'kill-buffer-hook #'+javascript|cleanup-tide-processes nil t))
|
|
|
|
|
2019-03-18 01:08:59 -04:00
|
|
|
(define-key tide-mode-map [remap +lookup/documentation] #'tide-documentation-at-point)
|
|
|
|
|
2018-12-23 23:54:27 -05:00
|
|
|
(map! :localleader
|
|
|
|
:map tide-mode-map
|
|
|
|
"R" #'tide-restart-server
|
|
|
|
"f" #'tide-reformat
|
|
|
|
"rs" #'tide-rename-symbol
|
|
|
|
"roi" #'tide-organize-imports))
|
2018-04-19 03:59:58 -04:00
|
|
|
|
|
|
|
|
2018-04-22 23:56:29 -04:00
|
|
|
(def-package! xref-js2
|
|
|
|
:when (featurep! :feature lookup)
|
2018-09-08 18:36:24 -04:00
|
|
|
:after (:or js2-mode rjsx-mode)
|
|
|
|
:config
|
|
|
|
(set-lookup-handlers! '(js2-mode rjsx-mode)
|
|
|
|
:xref-backend #'xref-js2-xref-backend))
|
2018-04-22 23:56:29 -04:00
|
|
|
|
|
|
|
|
2018-04-19 03:59:58 -04:00
|
|
|
(def-package! js2-refactor
|
2018-09-04 03:04:00 +02:00
|
|
|
:hook ((js2-mode rjsx-mode) . js2-refactor-mode)
|
2018-09-09 23:15:55 -04:00
|
|
|
:config
|
|
|
|
(when (featurep! :feature evil +everywhere)
|
|
|
|
(let ((js2-refactor-mode-map (evil-get-auxiliary-keymap js2-refactor-mode-map 'normal t t)))
|
|
|
|
(js2r-add-keybindings-with-prefix (format "%s r" doom-localleader-key)))))
|
2018-04-19 03:59:58 -04:00
|
|
|
|
2017-02-19 18:57:16 -05:00
|
|
|
|
2017-09-28 01:55:47 -07:00
|
|
|
(def-package! eslintd-fix
|
2018-05-25 00:46:11 +02:00
|
|
|
:commands eslintd-fix
|
2018-04-20 02:46:33 -04:00
|
|
|
:config
|
2018-05-14 20:55:55 +02:00
|
|
|
(defun +javascript|set-flycheck-executable-to-eslint ()
|
|
|
|
(setq flycheck-javascript-eslint-executable eslintd-fix-executable))
|
|
|
|
(add-hook 'eslintd-fix-mode-hook #'+javascript|set-flycheck-executable-to-eslint))
|
2017-09-28 01:55:47 -07:00
|
|
|
|
2017-03-23 15:47:07 -04:00
|
|
|
|
2018-05-25 00:46:11 +02:00
|
|
|
;; `skewer-mode'
|
2018-12-23 23:54:27 -05:00
|
|
|
(map! :localleader
|
|
|
|
:prefix "s"
|
|
|
|
(:after skewer-mode
|
2018-05-25 00:46:11 +02:00
|
|
|
:map skewer-mode-map
|
2018-12-23 23:54:27 -05:00
|
|
|
"E" #'skewer-eval-last-expression
|
|
|
|
"e" #'skewer-eval-defun
|
|
|
|
"f" #'skewer-load-buffer)
|
2018-04-20 02:44:04 -04:00
|
|
|
|
2018-05-25 00:46:11 +02:00
|
|
|
(:after skewer-css
|
|
|
|
:map skewer-css-mode-map
|
2018-12-23 23:54:27 -05:00
|
|
|
"e" #'skewer-css-eval-current-declaration
|
|
|
|
"r" #'skewer-css-eval-current-rule
|
|
|
|
"b" #'skewer-css-eval-buffer
|
|
|
|
"c" #'skewer-css-clear-all)
|
2017-03-24 15:01:03 -04:00
|
|
|
|
2018-05-25 00:46:11 +02:00
|
|
|
(:after skewer-html
|
|
|
|
:map skewer-html-mode-map
|
2018-12-23 23:54:27 -05:00
|
|
|
"e" #'skewer-html-eval-tag))
|
2017-03-24 15:01:03 -04:00
|
|
|
|
2018-09-07 22:08:11 -04:00
|
|
|
|
2018-08-31 23:10:23 +02:00
|
|
|
;; `npm-mode'
|
2018-09-02 17:18:31 +02:00
|
|
|
(map! :after npm-mode
|
|
|
|
:localleader
|
2018-12-23 23:54:27 -05:00
|
|
|
:map npm-mode-keymap
|
|
|
|
:prefix "n"
|
|
|
|
"n" #'npm-mode-npm-init
|
|
|
|
"i" #'npm-mode-npm-install
|
|
|
|
"s" #'npm-mode-npm-install-save
|
|
|
|
"d" #'npm-mode-npm-install-save-dev
|
|
|
|
"u" #'npm-mode-npm-uninstall
|
|
|
|
"l" #'npm-mode-npm-list
|
|
|
|
"r" #'npm-mode-npm-run
|
|
|
|
"v" #'npm-mode-visit-project-file)
|
2017-03-24 15:01:03 -04:00
|
|
|
|
2018-09-07 22:08:11 -04:00
|
|
|
|
2016-04-23 22:08:46 -04:00
|
|
|
;;
|
2017-01-16 23:30:37 -05:00
|
|
|
;; Projects
|
2016-05-20 16:55:32 -04:00
|
|
|
|
2018-08-31 02:51:40 +02:00
|
|
|
(def-project-mode! +javascript-npm-mode
|
2018-12-23 16:01:37 -05:00
|
|
|
:modes (html-mode css-mode web-mode typescript-mode js2-mode rjsx-mode json-mode markdown-mode)
|
2018-08-31 02:51:40 +02:00
|
|
|
:when (locate-dominating-file default-directory "package.json")
|
2018-08-31 23:10:23 +02:00
|
|
|
:add-hooks (+javascript|add-node-modules-path npm-mode))
|
2018-08-31 02:51:40 +02:00
|
|
|
|
|
|
|
(def-project-mode! +javascript-gulp-mode
|
|
|
|
:when (locate-dominating-file default-directory "gulpfile.js"))
|