1
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

[Emacs] Typescript+JSXをなるべくweb-modeで編集するための設定

Last updated at Posted at 2020-09-10

はじめに

ReactやTypescriptなどについて調べていると,この界隈にはVSCodeユーザが多いように感じられます.Emacsユーザにとっては少し肩身が狭いです.

Emacsを使った React + JSX の開発において,Javascriptを使う場合はrjsx-modejs-modeでおおむね事足ります.しかし,Typescriptを使う場合は,このような既存のモードでは少し不十分です.

私はこれまでrjsx-modeweb-modetypescript-modeなどを切り替えてやりくりしていましたが,これは一筋縄では行きません.例えば,rjsx-modeで引数の型指定をした部分から先がすべてエラーになったり,web-modeに切り替えた途端にカラーテーマが適用されなくなるようなことが起きてしまいます.

そこで,Typescript + JSXの編集をなるべくweb-modeで行うことができるようにするべく,その設定を行いました.

なお,私の環境ではカラーテーマをtango-darkとしています.その他のテーマでは,もしかすると必要に応じて最後の設定を直す必要があるかもしれません.

設定内容

~/.emacs.d/init.elに以下の設定を追記しました.

~/.emacs.d/init.el
;; web-mode settings for TSX
(add-to-list 'auto-mode-alist '("\\.[jt]sx\\'" . web-mode))
(defun custom-web-mode-hook ()
  (setq web-mode-attr-indent-offset nil)
  (setq web-mode-markup-indent-offset 2)
  (setq web-mode-css-indent-offset 4)
  (setq web-mode-code-indent-offset 4)
  (setq indent-tabs-mode nil)
  (setq tab-width 2)
  (setq web-mode-enable-current-element-highlight t)
  (let ((case-fold-search nil))
    (highlight-regexp "\\_<number\\|string\\|boolean\\|enum\\|unknown\\|any\\|void\\|null\\|undefined\\|never\\|object\\|symbol\\_>" 'font-lock-type-face)))
(add-hook 'web-mode-hook 'custom-web-mode-hook)
;; Inherit colors from font-lock
(custom-set-faces
 '(web-mode-doctype-face
   ((t :inherit font-lock-doc-face)))
 '(web-mode-html-tag-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-html-attr-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-html-attr-value-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-server-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-javascript-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-json-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-error-face
   ((t :inherit font-lock-warning-face)))
 '(web-mode-current-element-highlight-face
   ((t :inherit font-lock-builtin-face)))
 '(web-mode-html-tag-bracket-face
   ((t :inherit font-lock-negation-char-face)))
 '(web-mode-block-delimiter-face
   ((t :inherit font-lock-negation-char-face)))
 '(web-mode-javascript-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-json-key-face
   ((t :inherit font-lock-keyword-face)))
 '(web-mode-json-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-keyword-face
   ((t :inherit font-lock-keyword-face)))
 '(web-mode-param-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-preprocessor-face
   ((t :inherit font-lock-preprocessor-face)))
 '(web-mode-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-type-face
   ((t :inherit font-lock-type-face)))
 '(web-mode-variable-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-function-call-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-function-name-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-warning-face
   ((t :inherit font-lock-warning-face)))
 '(web-mode-css-color-face
   ((t :inherit font-lock-reference-face)))
 '(web-mode-css-rule-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-css-pseudo-class-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-css-at-rule-face
   ((t :inherit font-lock-keyword-face))))

説明

web-modeの有効化

まず,以下の設定によって.jsxもしくは.tsxのファイルを開いたときweb-modeを有効にしています.

(add-to-list 'auto-mode-alist '("\\.[jt]sx\\'" . web-mode))

インデントの設定

以下はインデントの設定です.rjsx-modeと近く※ なるように,JSXのみインデント幅が小さくなるように設定しています.

(※ あなたのEmacsでは,rjsx-modeのインデント幅が違うかもしれません)

  (setq web-mode-attr-indent-offset nil)
  (setq web-mode-markup-indent-offset 2)
  (setq web-mode-css-indent-offset 4)
  (setq web-mode-code-indent-offset 4)
  (setq indent-tabs-mode nil)
  (setq tab-width 2)

対応するタグをハイライトする設定

以下は,JSXを編集する際に対応するタグをハイライトするための設定です.ハイライトされる色 (face) は,web-mode-current-element-highlight-faceです.最後に他の色と合わせて設定しています.

  (setq web-mode-enable-current-element-highlight t)

型をハイライトする設定

以下は,Typescriptの型をハイライトするための設定です.正規表現の部分を編集することで,ハイライトする型を増やすことができます.ハイライトされる色 (face) は,font-lock-type-faceとしています.

なお,バッファ内で一致した単語をすべてハイライトするため,コメントに含まれるものなどであってもハイライトされます.ただし,case-fold-searchをセットしていることから,大文字・小文字は区別されます.

  (let ((case-fold-search nil))
    (highlight-regexp "\\_<number\\|string\\|boolean\\|enum\\|unknown\\|any\\|void\\|null\\|undefined\\|never\\|object\\|symbol\\_>" 'font-lock-type-face))

カラーテーマを(おおむね)適用する設定

以下は,使用中のカラーテーマの色(正確には,font-lockの色)をおおむね適用するための設定です.web-modeが定義する色 (face) の一部を,font-lockが定義する色 (face) に書き換えています.

なお,web-modefont-lockが定義する色 (face) は,M-x list-faces-displayで確認することができます.また,バッファ中のカーソルがある部分の色 (face) は,C-u C-x =で確認することができます.これらの情報は,以下の設定を変更する上で,とても役に立つでしょう.

(custom-set-faces
 '(web-mode-doctype-face
   ((t :inherit font-lock-doc-face)))
 '(web-mode-html-tag-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-html-attr-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-html-attr-value-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-server-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-javascript-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-json-comment-face
   ((t :inherit font-lock-comment-face)))
 '(web-mode-error-face
   ((t :inherit font-lock-warning-face)))
 '(web-mode-current-element-highlight-face
   ((t :inherit font-lock-builtin-face)))
 '(web-mode-html-tag-bracket-face
   ((t :inherit font-lock-negation-char-face)))
 '(web-mode-block-delimiter-face
   ((t :inherit font-lock-negation-char-face)))
 '(web-mode-javascript-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-json-key-face
   ((t :inherit font-lock-keyword-face)))
 '(web-mode-json-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-keyword-face
   ((t :inherit font-lock-keyword-face)))
 '(web-mode-param-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-preprocessor-face
   ((t :inherit font-lock-preprocessor-face)))
 '(web-mode-string-face
   ((t :inherit font-lock-string-face)))
 '(web-mode-type-face
   ((t :inherit font-lock-type-face)))
 '(web-mode-variable-name-face
   ((t :inherit font-lock-variable-name-face)))
 '(web-mode-function-call-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-function-name-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-warning-face
   ((t :inherit font-lock-warning-face)))
 '(web-mode-css-color-face
   ((t :inherit font-lock-reference-face)))
 '(web-mode-css-rule-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-css-pseudo-class-face
   ((t :inherit font-lock-function-name-face)))
 '(web-mode-css-at-rule-face
   ((t :inherit font-lock-keyword-face))))
1
3
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?