register_simp_attr

register_simp_attr は、simp タクティクで使うルールセットとタグを登録することができるコマンドで、simp ラッパとして新しいタクティクを作るのに使うことができます。

注意

このページの内容は ボタンから Lean 4 Web で実行することができません。

使用例

以下では、型クラスによって導入される記法を定義に展開するタクティクを自作する例を示します。

このコマンドは属性タグを作るので、まず別のファイルを作って以下のように書き込みます。ファイル名は何でも良いのですが、仮に RegisterSimpAttrLib.lean であるとします。

-- ## RegisterSimpAttrLib の内容

import Lean

/-- `+`や`≤`など、演算子や記法を定義に展開するためのルールを登録する -/
register_simp_attr notation_simp

その後、このファイルをインポートすれば @[notation_simp] タグとルールセットが利用可能になります。以下のように、macro_rules コマンドを使用すれば simp ラッパを作成することができます。

import LeanByExample.Declarative.RegisterSimpAttrLib -- インポートで有効になる

section
  open Lean Meta Parser.Tactic Elab.Tactic

  /-- `+`や`≤`など、演算子や記法を定義に展開する -/
  syntax (name := notation_simp_stx) "notation_simp" (simpArgs)? (location)? : tactic

  macro_rules
  | `(tactic| notation_simp $[[$simpArgs,*]]? $[at $location]?) =>
    let args := simpArgs.map (·.getElems) |>.getD #[]
  `(tactic| simp only [notation_simp, $args,*] $[at $location]?)
end

これで、たとえば以下のように使用することができます。

example {n m: Nat} (h : n < m) : n + 1 ≤ m := by
  -- notation_simp を使わない場合の方法。
  -- これを `(· < ·)` を展開したいときに毎回書く。
  dsimp [(· < ·), Nat.lt] at h
  assumption

@[notation_simp]
theorem Nat.lt_def (n m : Nat) : n < m ↔ (n + 1) ≤ m := by rfl

example {n m: Nat} (h : n < m) : n + 1 ≤ m := by
  -- これだけで展開が行えるようになった!
  notation_simp at h
  assumption

notation_simp? タクティクも自作するには、次のようにすればできます。

section
  open Lean Meta Parser.Tactic Elab.Tactic

  @[inherit_doc notation_simp_stx]
  syntax (name := notation_simp?) "notation_simp?" (simpArgs)? (location)? : tactic

  macro_rules
  | `(tactic| notation_simp? $[[$simpArgs,*]]? $[at $location]?) =>
    let args := simpArgs.map (·.getElems) |>.getD #[]
  `(tactic| simp? only [notation_simp, $args,*] $[at $location]?)
end

/- info: Try this: simp only [Nat.lt_def] at h -/
example {n m: Nat} (h : n < m) : n + 1 ≤ m := by
  notation_simp? at h
  assumption