38.6 ユーザー定義の“Things”とナビゲーション

バッファーにおいて関数やクラスの定義、ステートメント、コードブロック、文字列、コメント等といった特定のthings(物、オブジェクト)を識別して探せれば便利なときが多々あります。Emacsではどんな種類のtree-sitterノードが“thing”に相当するかをユーザーが定義できます。これにより次の関数へのジャンプ、ポイント位置にあるコードブロックのマーク、2つの関数の引数の入れ替えといった便利な機能が有効になります。

Emacsの“things”機能はtree-sitterのパターンマッチング機能とは独立した機能であり、パワーは比較的劣るもののパースツリーのナビゲーションと横断にはより適しています。

thingsの定義にはtreesit-thing-settings、定義したthingsの述語の取得はtreesit-thing-definition、あるthingsが定義済みかテストするにはtreesit-thing-defined-pを使うことができます。

Variable: treesit-thing-settings

これは言語それぞれにたいするthing定義のalistである。各エントリーのキーは言語シンボル、値は(thing pred)という形式のthing定義のリスト。ここでthingdefunsexpsentenceのようにthingを表すシンボル、predはこのthingがtree-sitterの何の種類のノードかを指定する。

predはノードのタイプにマッチするregexp文字列、ノードを引数としてそのノードがthingとして適格かを示すブーリアン値をリターンする関数、あるいは(regexp . fn)というコンスでもよい。これは正規表現regexpと関数fnからなるコンスであり、ノードはregexpとthingとしての適格性を調べるfnの両方を満足する必要がある。

predは再帰的に定義することもできる。(or pred…)と記述すると、predのいずれかが満たされればノードがthingとして適格であることを意味する。(not pred)と記述すれば、predが満たされないことでノードが適格であることを意味する。

最後にpredはこのリスト内で定義された他のpredを参照できる。たとえば(or sexp sentence)はこのあist内の他のルールとして定義されたsexpsentenceいずれかのthingであることを定義している。

以下はC/C++にたいするtreesit-thing-settingsの例:

((c
  (defun "function_definition")
  (sexp (not "[](),[{}]"))
  (comment "comment")
  (string "raw_string_literal")
  (text (or comment string)))
 (cpp
  (defun ("function_definition" . cpp-ts-mode-defun-valid-p))
  (defclass "class_specifier")
  (comment "comment")))

これは教示目的のために修正された例であって、C/C++のモードが実際にthingを定義する方法とは異なることに注意。

このセクションの残りの部分では、thing定義の恩恵を受ける関数をいくつか紹介します。以下の関数の他にもthing機能を利用する、たとえばtreesit-search-forwardtreesit-induce-sparse-tree等のツリー横断関数のような他の関数も別の場所で紹介されています。ノードの取得を参照してください。

Function: treesit-node-match-p node thing &optional ignore-missing

この関数はnodethingかどうかをチェックする。

nodethingなら非nil、それ以外はnilをリターンする。利便性のためにnodenilなら、この関数は単にnilをリターンする。

thingdefunのようなthingシンボル、あるいは"function_definition"(or comment string)のように単にthingを定義する述語でもよい。

thingが未定義や不正な場合には、デフォルトではこの関数はtreesit-invalid-predicateエラーをシグナルする。しかしこの関数はignore-missingtなら、thingが未定義でもエラーをシグナルせず単にnilをリターンする。しかしthingが不正な述語の場合には、依然としてエラーをシグナルする。

Function: treesit-thing-prev position thing

この関数はpositionの前にある指定されたthingであるような最初のノードをリターンする。そのようなノードがなければnilをリターンする。ノードがリターンされた場合には、ノード終端位置はposition以前であることが保証されている。言い換えると、この関数がpositionを取り囲むようなノードをリターンすることはあり得ない

繰り返しになるが、thingにはシンボルか述語のいずれかを指定できる。

Function: treesit-thing-next position thing

この関数はtreesit-thing-prevと似ているがpositionとりにあるthingであるような最初のノードだけをリターンする点が異なる。ノードがリターンされた場合には、ノード先頭位置はposition以降であることが保証されている。

Function: treesit-navigate-thing position arg side thing &optional tactic

この関数はtreesit-thing-prevtreesit-thing-next上に構築されており、ナビゲーションコマンドにおいて役に立つと思われる機能を提供する。この関数はpositionからarg個のthingインスタンスを横断して移動した後の位置をリターンする。横断するだけの十分なthingが存在しなければnilをリターンする。この関数はポイントを移動しない。

正のargthingのその個数のインスタンス分前方への移動、負のargは後方への移動を意味する。この関数はsidebegならthingの先頭で停止、endならthingの終端で停止する。

treesit-thing-prevの場合のようにthingtreesit-thing-settingsで定義されたthingシンボル、あるいは述語を指定できる。

tacticはこの関数がthing間を移動する方法を決定する。nestedtop-levelrestrictednilのいずれか。nestednilは通常のネストされたナビゲーションを意味する。まず兄弟を横断しての移動を試み、カレントレベルに兄弟がなければ親に移動してその兄弟を横断、のように移動する。top-levelはネストされたthingを無視してトップレベルのthingだけを横断した移動、restrictedpositionを囲い込むようなthingがあればその中に制限された移動を意味する。このtactic(戦術)はカレントネストレベルに留まって上位に移動したくないようなコマンドにたいして有用。

Function: treesit-thing-at position thing &optional strict

この関数はpositionを囲い込むthingであるような最小のノードをリターンする。そのようなノードがなければnilをリターンする。

リターンされるノードはpositionを囲い込んでいなければならない。すなわちノード開始位置がposition以前、ノード終端位置がposition以降であること。

strictが非nilなら、この関数は厳格な比較を使用する。すなわち開始位置はpositionより厳密に前、終端位置はpositionより厳密に後でなければならない。

thingtreesit-thing-settingsで定義されたthingシンボル、または述語のいずれか。

便利なラッパー関数もあります。treesit-beginning-of-thingはthing先頭にポイントを移動、treesit-end-of-thingはthing終端にポイントを移動、treesit-thing-at-pointはポイント位置にあるthingをリターンします。

treesit-beginning-of-defuntreesit-end-of-defuntreesit-defun-at-pointのように、(treesit-defun-type-regexpのフォールバックとしての)defun定義使用に特化したdefunコマンドもあります。さらにこれらの関数は移動の戦術としてtreesit-defun-tacticを使用します。これらの詳細については別のセクションで説明します(Tree-sitterとのメジャーモード開発を参照)。


This page has generated for branch:work/emacs-30_69b16e5c63840479270d32f58daea923fe725b90, commit:8c196e027afcda4529432b01ae733033b6ca1270 to check Japanese translation.