emacs/lisp/progmodes/ts-mode.el
Yuan Fu 170924e945
Fix js/ts tree-sitter template_string font-lock
* lisp/progmodes/js.el (js--treesit-settings): Fontify
template_strings with js--fontify-template-string.
(js--fontify-template-string): New function.
(js--json-treesit-settings): Add missing :feature flag.
* lisp/progmodes/ts-mode.el (ts-mode--settings): Fontify
template_strings with js--fontify-template-string.
2022-10-17 01:48:50 -07:00

367 lines
11 KiB
EmacsLisp

;;; ts-mode.el --- tree sitter support for TypeScript -*- lexical-binding: t; -*-
;; Copyright (C) 2022 Free Software Foundation, Inc.
;; Author : Theodor Thornhill <theo@thornhill.no>
;; Maintainer : Theodor Thornhill <theo@thornhill.no>
;; Created : October 2022
;; Keywords : typescript tsx languages tree-sitter
;; This file is part of GNU Emacs.
;; 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/>.
;;; Code:
(require 'treesit)
(require 'rx)
(require 'js)
(defcustom ts-mode-indent-offset 2
"Number of spaces for each indentation step in `ts-mode'."
:type 'integer
:safe 'integerp
:group 'typescript)
(defvar ts-mode--syntax-table
(let ((table (make-syntax-table)))
;; Taken from the cc-langs version
(modify-syntax-entry ?_ "_" table)
(modify-syntax-entry ?$ "_" table)
(modify-syntax-entry ?\\ "\\" table)
(modify-syntax-entry ?+ "." table)
(modify-syntax-entry ?- "." table)
(modify-syntax-entry ?= "." table)
(modify-syntax-entry ?% "." table)
(modify-syntax-entry ?< "." table)
(modify-syntax-entry ?> "." table)
(modify-syntax-entry ?& "." table)
(modify-syntax-entry ?| "." table)
(modify-syntax-entry ?` "\"" table)
(modify-syntax-entry ?\240 "." table)
table)
"Syntax table for `ts-mode'.")
(defvar ts-mode--indent-rules
`((tsx
((node-is "}") parent-bol 0)
((node-is ")") parent-bol 0)
((node-is "]") parent-bol 0)
((node-is ">") parent-bol 0)
((node-is ".")
parent-bol ,ts-mode-indent-offset)
((parent-is "ternary_expression")
parent-bol ,ts-mode-indent-offset)
((parent-is "named_imports")
parent-bol ,ts-mode-indent-offset)
((parent-is "statement_block")
parent-bol ,ts-mode-indent-offset)
((parent-is "type_arguments")
parent-bol ,ts-mode-indent-offset)
((parent-is "variable_declarator")
parent-bol ,ts-mode-indent-offset)
((parent-is "arguments")
parent-bol ,ts-mode-indent-offset)
((parent-is "array")
parent-bol ,ts-mode-indent-offset)
((parent-is "formal_parameters")
parent-bol ,ts-mode-indent-offset)
((parent-is "template_substitution")
parent-bol ,ts-mode-indent-offset)
((parent-is "object_pattern")
parent-bol ,ts-mode-indent-offset)
((parent-is "object")
parent-bol ,ts-mode-indent-offset)
((parent-is "object_type")
parent-bol ,ts-mode-indent-offset)
((parent-is "enum_body")
parent-bol ,ts-mode-indent-offset)
((parent-is "arrow_function")
parent-bol ,ts-mode-indent-offset)
((parent-is "parenthesized_expression")
parent-bol ,ts-mode-indent-offset)
;; TSX
((parent-is "jsx_opening_element")
parent ,ts-mode-indent-offset)
((node-is "jsx_closing_element") parent 0)
((parent-is "jsx_element")
parent ,ts-mode-indent-offset)
((node-is "/") parent 0)
((parent-is "jsx_self_closing_element")
parent ,ts-mode-indent-offset)
(no-node parent-bol 0)))
"Tree-sitter indent rules.")
(defvar ts-mode--settings
(treesit-font-lock-rules
:language 'tsx
:override t
:feature 'basic
'(((identifier) @font-lock-constant-face
(:match "^[A-Z_][A-Z_\\d]*$" @font-lock-constant-face))
(nested_type_identifier
module: (identifier) @font-lock-type-face)
(type_identifier) @font-lock-type-face
(predefined_type) @font-lock-type-face
(new_expression
constructor: (identifier) @font-lock-type-face)
(function
name: (identifier) @font-lock-function-name-face)
(function_declaration
name: (identifier) @font-lock-function-name-face)
(method_definition
name: (property_identifier) @font-lock-function-name-face)
(variable_declarator
name: (identifier) @font-lock-function-name-face
value: [(function) (arrow_function)])
(variable_declarator
name: (array_pattern
(identifier)
(identifier) @font-lock-function-name-face)
value: (array (number) (function)))
(assignment_expression
left: [(identifier) @font-lock-function-name-face
(member_expression
property: (property_identifier) @font-lock-function-name-face)]
right: [(function) (arrow_function)])
(call_expression
function:
[(identifier) @font-lock-function-name-face
(member_expression
property: (property_identifier) @font-lock-function-name-face)])
(variable_declarator
name: (identifier) @font-lock-variable-name-face)
(enum_declaration (identifier) @font-lock-type-face)
(enum_body (property_identifier) @font-lock-type-face)
(enum_assignment name: (property_identifier) @font-lock-type-face)
(assignment_expression
left: [(identifier) @font-lock-variable-name-face
(member_expression
property: (property_identifier) @font-lock-variable-name-face)])
(for_in_statement
left: (identifier) @font-lock-variable-name-face)
(arrow_function
parameter: (identifier) @font-lock-variable-name-face)
(arrow_function
parameters:
[(_ (identifier) @font-lock-variable-name-face)
(_ (_ (identifier) @font-lock-variable-name-face))
(_ (_ (_ (identifier) @font-lock-variable-name-face)))])
(pair key: (property_identifier) @font-lock-variable-name-face)
(pair value: (identifier) @font-lock-variable-name-face)
(pair
key: (property_identifier) @font-lock-function-name-face
value: [(function) (arrow_function)])
(property_signature
name: (property_identifier) @font-lock-variable-name-face)
((shorthand_property_identifier) @font-lock-variable-name-face)
(pair_pattern
key: (property_identifier) @font-lock-variable-name-face)
((shorthand_property_identifier_pattern)
@font-lock-variable-name-face)
(array_pattern (identifier) @font-lock-variable-name-face)
(jsx_opening_element
[(nested_identifier (identifier)) (identifier)]
@font-lock-function-name-face)
(jsx_closing_element
[(nested_identifier (identifier)) (identifier)]
@font-lock-function-name-face)
(jsx_self_closing_element
[(nested_identifier (identifier)) (identifier)]
@font-lock-function-name-face)
(jsx_attribute (property_identifier) @font-lock-constant-face)
[(this) (super)] @font-lock-keyword-face
[(true) (false) (null)] @font-lock-constant-face
(regex pattern: (regex_pattern)) @font-lock-string-face
(number) @font-lock-constant-face
(string) @font-lock-string-face
(template_string) @js--fontify-template-string
(template_substitution
["${" "}"] @font-lock-constant-face)
["!"
"abstract"
"as"
"async"
"await"
"break"
"case"
"catch"
"class"
"const"
"continue"
"debugger"
"declare"
"default"
"delete"
"do"
"else"
"enum"
"export"
"extends"
"finally"
"for"
"from"
"function"
"get"
"if"
"implements"
"import"
"in"
"instanceof"
"interface"
"keyof"
"let"
"namespace"
"new"
"of"
"private"
"protected"
"public"
"readonly"
"return"
"set"
"static"
"switch"
"target"
"throw"
"try"
"type"
"typeof"
"var"
"void"
"while"
"with"
"yield"
] @font-lock-keyword-face
(comment) @font-lock-comment-face
))
"Tree-sitter font-lock settings.")
(defvar ts-mode--defun-type-regexp
(rx (or "class_declaration"
"method_definition"
"function_declaration"
"lexical_declaration"))
"Regular expression that matches type of defun nodes.
Used in `ts-mode--beginning-of-defun' and friends.")
(defun ts-mode--beginning-of-defun (&optional arg)
"Tree-sitter `beginning-of-defun' function.
ARG is the same as in `beginning-of-defun."
(let ((arg (or arg 1)))
(if (> arg 0)
;; Go backward.
(while (and (> arg 0)
(treesit-search-forward-goto
ts-mode--defun-type-regexp 'start nil t))
(setq arg (1- arg)))
;; Go forward.
(while (and (< arg 0)
(treesit-search-forward-goto
ts-mode--defun-type-regexp 'start))
(setq arg (1+ arg))))))
(defun ts-mode--end-of-defun (&optional arg)
"Tree-sitter `end-of-defun' function.
ARG is the same as in `end-of-defun."
(let ((arg (or arg 1)))
(if (< arg 0)
;; Go backward.
(while (and (< arg 0)
(treesit-search-forward-goto
ts-mode--defun-type-regexp 'end nil t))
(setq arg (1+ arg)))
;; Go forward.
(while (and (> arg 0)
(treesit-search-forward-goto
ts-mode--defun-type-regexp 'end))
(setq arg (1- arg))))))
;;;###autoload
(add-to-list 'auto-mode-alist '("\\.ts\\'" . ts-mode))
;;;###autoload
(add-to-list 'auto-mode-alist '("\\.tsx\\'" . ts-mode))
;;;###autoload
(define-derived-mode ts-mode prog-mode "TypeScript"
"Major mode for editing TypeScript."
:group 'typescript
:syntax-table ts-mode--syntax-table
(cond
((and (treesit-can-enable-p)
(treesit-language-available-p 'tsx))
;; Comments
(setq-local comment-start "// ")
(setq-local comment-start-skip "\\(?://+\\|/\\*+\\)\\s *")
(setq-local comment-end "")
(setq-local treesit-simple-indent-rules ts-mode--indent-rules)
(setq-local indent-line-function #'treesit-indent)
(setq-local beginning-of-defun-function #'ts-mode--beginning-of-defun)
(setq-local end-of-defun-function #'ts-mode--end-of-defun)
(unless font-lock-defaults
(setq font-lock-defaults '(nil t)))
(setq-local treesit-font-lock-settings ts-mode--settings)
(setq treesit-font-lock-feature-list '((basic)))
(treesit-font-lock-enable))
(t
(message "Tree sitter for TypeScript isn't available, defaulting to js-mode")
(js-mode))))
(provide 'ts-mode)
;;; ts-mode.el ends here