Skip to content

citizen428/transient-showcase

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

35 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Transient Showcase

Code examples for interactive explanations of transient.

This guide assumes you have minimal knowledge of Emacs, some programming experience in elisp and non-lisp languages, and have at least seen screenshots of magit.

How to use

There are two ways:

  • Open this file in Emacs and run examples as literate org.
  • Install the package to run commands and read their source. Start with the ts-showcase command.

Using as literate org document

If you open this file in Emacs, it will switch to Org mode and you can run individual source blocks with org-babel-execute-src-blk on the block.

Using as an installable package

If you install the package, you can read source for each example with the normal. describe-function command. All commands are under ts-* prefix. Somewhat useful suffixes are under ts-suffix-* while less useful ones under ts--suffix-*. They will come in handly when you are developing new applications.

Installations for straight and elpaca:

;; using straight use-package with custom recipe
(straight-use-package
 '(transient-showcase
   :type git :host github :repo "positron-solutions/transient-showcase")
 :demand t)

;; using elpaca (recommended to add a hash for reproducibility)
(elpaca-use-package
 (example :host github
          :repo "positron-solutions/transient-showcase")
 :demand t)

Note While this file is also the README for this repository, it’s not intended to be used copy-paste. Many links will only open in Emacs. Some definitions are included by refrence from Preludes

Packaging

This file is produced with org-babel-tangle. Package header and preludes included in the no-web block below:

<<package-header>>
<<wave-prelude>>
<<predicates-prelude>>
<<show-level-prelude>>
<<levels-prelude>>
<<print-args-prelude>>

Running Examples in Org Mode

This is a basic transient, using an anonymous lambda interactive command as its only suffix.

(transient-define-prefix ts-hello ()
  "Prefix that is minimal and uses an anonymous command suffix."
  [("s" "call suffix"
    (lambda ()
      (interactive)
      (message "Called a suffix")))])

;; First, use M-x org-babel-execute-src-blk to cause `ts-hello' to be defined
;; Second, M-x `eval-last-sexp' with your point at the end of the line below
;; (ts-hello)

After executing the block above, you can execute-extended-command (M-x) and select ts-hello to show this transient. All transient prefixes are also commands that show up in (M-x)

Note If the example above is hard to read, review some elisp syntax and typical forms.

Contents

Terminology

Transient means temporary. Transient gets it’s name from the temporary keymap and the popup UI for displaying that keymap. Emacs has a similar idea built-in with set-transient-map for a temporary high-precedence keymap.

Prefixes and Suffixes

The hello transient user input sequence is:

Prefix -> Suffix

  • The prefix is the command you invoke first, such as magit-dispatch
  • A suffix is a command displayed in the transient UI, such as magit-stage
    (magit-dispatch) ; same as pressing 'h' in magit-status buffer
        

The keymap and UI display is frequently referred to as “a transient”. “Prefix” and “a transient” are almost the same thing. Invoking a prefix will show a transient. They are inseparable ideas.

Conceptual similarity to Emacs prefix arguments

Setting prefix arguments with =universal-argument= (=C-u=) is a distinct, separate behavior that is part of Emacs.

With prefix arguments, you “call” commands with extra arguments, like you would a function.

A transient prefix can set some states and its suffix can then use these states to tweak its behavior. The difference is that within the lifecycle of a transient UI, and coordinating with transient’s state persistence, you can create much more complex input to your commands. You can use commands to construct phrases for other commands.

To see a short example of prefix arguments being used within a transient prefix, see the scope example.

Nesting Prefixes

A prefix can also be bound as a suffix, enabling nested prefixes. A user input sequence with nested transients might look like:

Prefix -> Sub-Prefix -> Sub-Prefix -> Suffix

For example, in the magit-dispatch transient (?), l for magit-log is a nested transient. b for all branches is the suffix command magit-log-all-branches.

See Flow Control for nested transient examples with both sub-prefixes and suffixes that do no exit.

Infix

Some suffixes need to hold state, toggling or storing an argument. Infixes are specialized suffixes to set and hold state. A user input sequence with infixes:

Prefix -> Infix -> Infix -> Suffix

See Infix examples to get a better idea.

Summary

  • Prefixes display the pop-up UI and bind the keymap.
  • Suffixes are commands bound within a prefix
  • Infixes are a specialized suffix for storing and setting state
  • A Suffix may be yet another Prefix, in which case the transient is nested

Declaring - Equivalent Forms

You can declare the same behavior 3-4 ways

  • Shorthand forms within transient-define-prefix macro allow shorthand binding of suffixes & commands or creation of infixes directly within the layout definition.
  • Macros for suffixes and infix definition streamline defining commands while also defining how they will behave in a layout.
  • Keyword arguments (:foo val1 :bar val2) are interpreted by the macros and used to set slots (OOP attributes) on prefix, group, and suffix objects. Similar forms for declaring suffixes can be used to modify them when declaring a layout. Very specific control over layouts also uses these forms.
    ;; slots & methods that can be set / overridden in children
    (describe-function transient-child)
          
  • Custom classes using EIEIO (basically elisp OOP) can change methods deeper in the implementation than you can reach with slots. describe-function is a quick way to look at the methods.
    ;; slots & methods that can be set / overridden in suffixes
    (describe-function transient-suffix)
          

    See the EIEIO Appendix for introduction to exploring EIEIO objects and classes.

The Shorthand form

Binding suffixes with the ("key" "description" suffix-or-command) form within a group is extremely common.

(transient-define-prefix ts-wave ()
  "Prefix that waves at the user"
  [("w" "wave" ts-suffix-wave)]) ; ts-suffix-wave is a simple command from wave-prelude

;; (ts-wave)

Note: Both commands and suffixes from transient-define-suffix can be used. It’s a good reason to use private--namespace style names for suffix actions since these commands don’t usually show up in (M-x) by default.

Keyword Arguments Style

You can customize the slot value (OOP attribute) of the transient, groups, and suffixes by adding extra :foo value style pairs.

Not all behaviors have a shorthand form, so as you use more behaviors, you will see more of the keyword argument style API. Here we use the :transient property, set to true, meaning the suffix won’t exit the transient.

(transient-define-prefix ts-wave-keyword-args ()
  "Prefix that waves at the user persistently."
  [("e" "wave eventually & stay" ts--wave-eventually :transient t)
   ("s" "wave surely & leave" ts--wave-surely :transient nil)])

;; (ts-wave-keyword-args)

Launch the command, wave several times (note timestamp update) and then exit with (C-g).

Macro Child Definition Style

The transient-define-suffix macro can help if you need to bind a command in multiple places and only override some properties for some prefixes. It makes the prefix definition more compact at the expense of a more verbose command.

(transient-define-suffix ts-suffix-wave-macroed ()
  "Prefix that waves with macro-defined suffix."
  :transient t
  :key "T"
  :description "wave from macro definition"
  (interactive)
  (message "Waves from a macro definition at: %s" (current-time-string)))
;; ts-suffix-wave-suffix defined above

(transient-define-prefix ts-wave-macro-defined ()
  "Prefix to wave using a macro-defined suffix"
  [(ts-suffix-wave-macroed)]) ; note, information moved from prefix to the suffix.

;; (ts-wave-macro-defined)

Overriding slots in the prefix definition

Even if you define a property via one of the macros, you can still override that property in the later prefix definition. The example below overrides the :transient, :description, and :key properties of the ts-suffix-wave suffix defined above:

(defun ts--wave-override ()
  "Vanilla command used to override suffix's commands."
  (interactive)
  (message "This suffix was overridden.  I am what remains."))

(transient-define-prefix ts-wave-overridden ()
  "Prefix that waves with overridden suffix behavior"
  [(ts-suffix-wave-macroed
    :transient nil
    :key "O"
    :description "wave overridingly"
    :command ts--wave-override)]) ; we overrode what the suffix even does

;; (ts-wave-overridden)

If you just list the key and symbol followed by properties, it is also a supported shorthand suffix form:

("wf" ts-suffix-wave :description "wave furiously")

Quoting Note for Vectors

Inside the [ ...vectors... ] in transient-define-prefix, you don’t need to quote symbols because in the vector, everything is a literal. When you move a shorthand style :property symbol out to the transient-define-suffix form, which is a list, you might need to quote the symbol as :property 'symbol.

Groups & Layouts

To define a transient, you need at least one group. Groups are vectors, delimited as [ ...group... ].

There is basic layout support and you can use it to collect or differentiate commands.

If you begin a group vector with a string, you get a group heading. Groups also support some properties. The group class also has a lot of information.

Descriptions

Very straightforward. Just make the first element in the vector a string or add a :description property, which can be a function.

In the prefix definition of suffixes, the second string is a description.

The :description key is applied last and therefore wins in ambiguous declarations.

(transient-define-prefix ts-layout-descriptions ()
  "Prefix with descriptions specified with slots."
  ["Let's Give This Transient a Title\n" ; yes the newline works
   ["Group One"
    ("wo" "wave once" ts-suffix-wave)
    ("wa" "wave again" ts-suffix-wave)]

   ["Group Two"
    ("ws" "wave some" ts-suffix-wave)
    ("wb" "wave better" ts-suffix-wave)]]

  ["Bad title" :description "Group of Groups"
   ["Group Three"
    ("k" "bad desc" ts-suffix-wave :description "key-value wins")
    ("n" ts-suffix-wave :description "no desc necessary")]
   [:description "Key Only Def"
    ("wt" "wave too much" ts-suffix-wave)
    ("we" "wave excessively" ts-suffix-wave)]])

;; (ts-layout-descriptions)

Dynamic Descriptions

Note: The property list style for dynamic descriptions is the same for both prefixes and suffixes. Add :description symbol-or-lambda-form to the group vector or suffix list.

 (transient-define-prefix ts-layout-dynamic-descriptions ()
   "Prefix that generate descriptions dynamically when transient is shown."
   ;; group using function-name to generate description
   [:description current-time-string
    ;; single suffix with dynamic description
    ("wa" ts-suffix-wave
     :description (lambda ()
                    (format "Wave at %s" (current-time-string))))]
   ;; group with anonymoous function generating description
   [:description (lambda ()
                   (format "Group %s" (org-id-new)))
                 ("wu" "wave uniquely" ts-suffix-wave)])

;; (ts-layout-dynamic-descriptions)

Errata

Note, the uuid is generated on every key input. Layout updates are fun. It does not also work when changing descriptions in the layout via hackery. 凸( ` ロ ´ )凸

Layouts

The default behavior treats groups a little differently depending on how they are nested. For most simple groupings, this is sufficient control.

Groups one on top of the other

Use a vector for each row.

(transient-define-prefix ts-layout-stacked ()
  "Prefix with layout that stacks groups on top of each other."
  ["Top Group" ("wt" "wave top" ts-suffix-wave)]
  ["Bottom Group" ("wb" "wave bottom" ts-suffix-wave)])

;; (ts-layout-stacked)

Groups side by side

Use a vector of vectors for columns.

(transient-define-prefix ts-layout-columns ()
  "Prefix with side-by-side layout."
  [["Left Group" ("wl" "wave left" ts-suffix-wave)]
   ["Right Group" ("wr" "wave right" ts-suffix-wave)]])

;; (ts-layout-columns)

Group on top of groups side by side

Vector on top of vector inside a vector.

(transient-define-prefix ts-layout-stacked-columns ()
  "Prefix with stacked columns layout."
  ["Top Group"
   ("wt" "wave top" ts-suffix-wave)]

  [["Left Group"
    ("wl" "wave left" ts-suffix-wave)]
   ["Right Group"
    ("wr" "wave right" ts-suffix-wave)]])

;; (ts-layout-stacked-columns)

*Note: Groups can have groups or suffixes, but not both. You can’t mix suffixes alongside groups in the same vector. The resulting transient will error when invoked.*

Empty strings make spaces

Groups that are empty or only space have no effect. This situation can happen with layouts that update dynamically. See dynamic layouts.

(transient-define-prefix ts-layout-spaced-out ()
  "Prefix lots of spacing for users to space out at."
  ["" ; cannot add another empty string because it will mix suffixes with groups
   ["Left Group"
    ""
    ("wl" "wave left" ts-suffix-wave)
    ("L" "wave lefter" ts-suffix-wave)
    ""
    ("bl" "wave bottom-left" ts-suffix-wave)
    ("z" "zone\n" zone)] ; the newline does pad

   [[]] ; empty vector will do nothing

   [""] ; vector with just empty line has no effect

   ;; empty group will be ignored
   ;; (useful for hiding in dynamic layouts)
   ["Empty Group\n"]

   ["Right Group"
    ""
    ("wr" "wave right" ts-suffix-wave)
    ("R" "wave righter" ts-suffix-wave)
    ""
    ("br" "wave bottom-right" ts-suffix-wave)]])

;; (ts-layout-spaced-out)

A Grid

So, you put columns into rows that are in columns and stuff like that. This can be achieved with or without explicit column settings.

(transient-define-prefix ts-layout-the-grid ()
  "Prefix with groups in a grid-like arrangement."

  [:description "The Grid\n" ; must use slot or macro is confused
   ["Left Column" ; note, no newline
    ("ltt" "left top top" ts-suffix-wave)
    ("ltb" "left top bottom" ts-suffix-wave)
    ""
    ("lbt" "left bottom top" ts-suffix-wave)
    ("lbb" "left bottom bottom" ts-suffix-wave)] ; note, no newline

   ["Right Column\n"
    ("rtt" "right top top" ts-suffix-wave)
    ("rtb" "right top bottom" ts-suffix-wave)
    ""
    ("rbt" "right bottom top" ts-suffix-wave)
    ("rbb" "right bottom bottom\n" ts-suffix-wave)]])

;; (ts-layout-the-grid)

Note, only transient-columns, not transient-column can act as a group of groups.

Manually setting group class

If you need to override the class that the transient-define-prefix macro would normally use.

(transient-define-prefix ts-layout-explicit-classes ()
  "Prefix with group class used to explicitly specify layout."
  [:class transient-row "Row"
          ("l" "wave left" ts-suffix-wave)
          ("r" "wave right" ts-suffix-wave)]
  [:class transient-column "Column"
          ("t" "wave top" ts-suffix-wave)
          ("b" "wave bottom" ts-suffix-wave)])

;; (ts-layout-explicit-classes)

Nesting & Flow Control

Many transients call other transients. This allows you to express similar behaviors as interactive commands that ask you for multiple arguments using the minibuffer.

Transient has more options for retaining some state across several transients, making it easier to compose commands and to retain intermediate states for rapidly achieving series of actions over similar inputs.

Single versus multiple commands

Sometimes you want to execute multiple commands without re-opening the transient. It’s the same idea as god mode or Evil repeat.

(transient-define-prefix ts-stay-transient ()
  "Prefix where some suffixes do not exit."
  ["Exit or Not?"

   ;; this suffix will not exit after calling sub-prefix
   ("we" "wave & exit" ts-wave-overridden)
   ("ws" "wave & stay" ts-wave :transient t)])

;; (ts-stay-transient)

Note, if ts-wave was used in both exit & stay, the :transient slot would be clobbered and we would only get one behavior. Beware of re-using the same object instances in the same layout. Move the :transient slot override between the two suffixes to see the change in behavior.

Nesting

Nesting is putting transients inside other transients, creating user-input sequences like:

Prefix -> Sub-Prefix -> Suffix

Binding a Sub-Prefix

This is the most simple way to create nesting.

(transient-define-prefix ts--simple-child ()
  ["Simple Child"
   ("wc" "wave childishly" ts-suffix-wave)])

(transient-define-prefix ts-simple-parent ()
  "Prefix that calls a child prefix."
  ["Simple Parent"
   ("w" "wave parentally" ts-suffix-wave)
   ("b" "become child" ts--simple-child)])

;; (ts--simple-child)
;; (ts-simple-parent)

Nesting with multiple commands

Declaring a nested prefix that “returns” to its parent has a convenient shorthand form.

(transient-define-prefix ts-simple-parent-with-return ()
  "Prefix with a child prefix that returns."
  ["Parent With Return"
   ("w" "wave parentally" ts-suffix-wave)
   ("b" "become child with return" ts--simple-child :transient t)])

;; Child does not "return" when called independently
;; (ts--simple-child)
;; (ts-simple-parent-with-return)

Setting up another transient manually

If you call (transient-setup 'transient-command-symbol), you will activate a replacement transient.

This form is useful if you want a command to perhaps load yet another transient in some situation. You may even just want to load the same transient with different context, such as passing in a new scope.

(transient-define-suffix ts-suffix-setup-child ()
  "A suffix that uses `transient-setup' to manually load another transient."
  (interactive)
  ;; note that it's usually during the post-command side of calling the
  ;; command that the actual work to set up the transient will occur.
  ;; This is an implementation detail because it depends if we are calling
  ;; `transient-setup' while already transient or not.
  (transient-setup 'ts--simple-child))

(transient-define-prefix ts-parent-with-setup-suffix ()
  "Prefix with a suffix that calls `transient-setup'."
  ["Simple Parent"
   ("wp" "wave parentally" ts-suffix-wave :transient t) ; remain transient

   ;; You may need to specify a different pre-command (the :transient) key
   ;; because we need to clean up this transient or create some conditions
   ;; to trigger the following transient correctly.  This example will
   ;; work with `transient--do-replace' or no custom pre-command

   ("bc" "become child" ts-suffix-setup-child :transient transient--do-replace)])

;; (ts-parent-with-setup-suffix)

Errata

This example should also work with the transient--do-recurse pre-command, but the child transient does not return. There is a difference in the behavior that should not depend on if the suffix is the prefix or just sets up the prefix. Possible bug.

Mixing Interactive

You can mix normal Emacs completion flows with transient UI’s.

See Interactive codes are listed in the Elisp manual.

Note, this also works when binding existing commands that recieve user input.

(transient-define-suffix ts--suffix-interactive-string (user-input)
  "An interactive suffix that obtains string input from the user."
  (interactive "sPlease just tell me what you want!: ")
  (message "I think you want: %s" user-input))

(transient-define-suffix ts--suffix-interactive-buffer-name (buffer-name)
  "An interactive suffix that obtains a buffer name from the user."
  (interactive "b")
  (message "You selected: %s" buffer-name))

(transient-define-prefix ts-interactive-basic ()
  "Prefix with interactive user input."
  ["Interactive Command Suffixes"
   ("s" "enter string" ts--suffix-interactive-string)
   ("b" "select buffer" ts--suffix-interactive-buffer-name)])

;; (ts-interactive-basic)

Early return

Sometimes you can complete your work without asking the user for more input. In the custom body for a prefix, if you decline to call transient-setup, then the command will just exit with no problems.

Below is a nested transient.

  • The body form of the nested child can return early without loading a new transient
  • The parent uses transient--do-recurse to make it’s child “return” to it
  • The “radiations” command in the child explicitly overrides this, using transient--do-exit so that it does not return to the parent
(defvar ts--complex nil "Show complex menu or not")

(transient-define-suffix ts--toggle-complex ()
  :transient t
  :description (lambda () (format "toggle complex: %s" ts--complex))
  (interactive)
  (setf ts--complex (not ts--complex))
  (message (propertize (concat "Complexity set to: "
                               (if ts--complex "true" "false"))
                       'face 'success)))

(transient-define-prefix ts-complex-messager ()
  "Prefix that sends complex messages, unles `ts--complex' is nil."
  ["Send Complex Messages"
   ("s" "snow people"
    (lambda () (interactive)
      (message (propertize "snow people! ☃" 'face 'success))))
   ("k" "kitty cats"
    (lambda () (interactive)
      (message (propertize "🐈 kitty cats! 🐈" 'face 'success))))
   ("r" "radiations"
    (lambda () (interactive)
      (message (propertize "Oh no! radiation! ☢" 'face 'success)))
    ;; radiation is dangerous!
    :transient transient--do-exit)]

  (interactive)
  ;; The command body either sets up the transient or simply returns
  ;; This is the "early return" we're talking about.
  (if ts--complex
      (transient-setup 'ts-complex-messager)
    (message "Simple and boring!")))

(transient-define-prefix ts-simple-messager ()
  "Prefix that toggles child behavior!"
  [["Send Message"
    ;; using `transient--do-recurse' causes suffixes in ts-child to perform
    ;; `transient--do-return' so that we come back to this transient.
    ("m" "message" ts-complex-messager :transient transient--do-recurse)]
   ["Toggle Complexity"
    ("t" ts--toggle-complex)]])

;; (ts-simple-messager)
;; does not "return" when called independently
;; (ts-complex-messager)

Pre-Commands Explained

The value in the :transient slot affects what state the body of your command will see and what will happen after your command, during the post-command.

The :transient slot holds a function called the “pre-command.” Before your suffix body forms run, the pre-command is called and creates the conditions that your suffix may use to, for example, prepare for reading variables that were set on infixes. If the pre-command calls transient-export then it will add to history.

In transient-define-prefix and transient-define-suffix, the t value is actually translated to transient--do-call or transient--do-recurse depending on the situation.

These functions set up some states so that post-command can figure out if it needs to exit, save values, or enter another transient, and what else to do while entering that new transient.

The official long manual has some more detail. These examples should prepare you to visualize the forms used in those explanations.

Warning!

Some of the trickiest bugs you can introduce will happen when using the following variables and functions at varying points in command lifecycles:

  • transient-current-command
  • transient--command
  • transient-current-prefix
  • transient--prefix
  • transient-args

During the pre-command and post-command, these can change. When you are overriding the pre-command, you may discover things such as the result of transient-args changing. Calling transient-setup may update things. Even if you call transient-args on on the specific transient, the results change during the lifecycle and depending on the pre-command.

In particular it seems like layout predicates should use transient--prefix while suffix bodies should use transient-current-prefix.

Not all pre-commands are compatible with all situations and suffixes!

Debugging

Errata

There’s definitely some edge cases that are unnecessarily complex for the use case. Think of how life was before transient--do-recurse.

Using & Managing States

There are several ways to create state. The flow control examples in the previous section mainly covered how to get from one command to the other. This section covers how to save values and then read them later, sometimes from a completely different transient. Coupled with custom infix types, you can create some seriously rich user expression.

To spark your imagination, here’s a non-exhaustive list of how to get data into your commands:

  • Interactive forms
  • Prefix arguments (C-u universal argument)
  • Setting the scope in transient-setup
  • Obtaining a scope in a custom transient-init-scope method
  • Default values in prefix definition
  • Saved values of infixes
  • Saved values in other infixes / prefixs with shared history-key
  • User-set infix values from the current or parent prefix
  • Ad-hoc values in regular defvar and defcustom etc
  • Reading values from another, perhaps distant prefix
  • Arguments passed into interactive commands to call them as normal elisp functions

The Magic of Transient

Using all of these mechanisms, you can enable users to rapidly construct complex command sentences, sentences with phrases. You can basically make a user interface as expressive as elisp.

A user input sequence like this:

Prefix -> Interactive -> Sub-Prefix -> Infix -> Suffix -> Suffix -> ...

Is basically the same as doing this in elisp:

(let ((input (Sub-Prefix (Prefix) (Interactive)))
      (infix (Infix))
  (suffix input infix)
  (suffix input infix)))

With history, you can remember lots of these states. This allows the user to quickly fire off lots of mostly completed partial expressions. They are scoped, so you can keep state over different contexts.

This is what is meant by “creating user interfaces as expressive as elisp.”

Because interactive forms and transients are both still just consuming linear user input, they ultimately have the same capabilities, but if you think in terms of partially constructed elisp expressions, you can do more than if the user has to enter in contextless commands over and over or write more commands while managing their own state in ad-hoc fashion.

Transient’s UI also provides greater awareness to the user of the current state. This makes it easier for the user to achieve the greater complexity that is intended, without remembering the command language you are designing for your application.

Infixes

Functions need arguments. Infixes are specialized suffixes with behavior defaults that make sense for setting and storing values for consumption in suffixes. It’s like passing arguments into the suffix. They also have support for persisting state across invocations and Emacs sessions.

Basic Infixes

Infix classes built-in all descend from transient-infix and can be seen clearly in the eieio-browse. View their slots and documentaiton with (describe-class transient-infix) etc. Here you can see what most infixes look like and how they behave.

;; infix defined with a macro
(transient-define-argument ts--exclusive-switches ()
  "This is a specialized infix for only selecting one of several values."
  :class 'transient-switches
  :argument-format "--%s-snowcone"
  :argument-regexp "\\(--\\(grape\\|orange\\|cherry\\|lime\\)-snowcone\\)"
  :choices '("grape" "orange" "cherry" "lime"))

(transient-define-prefix ts-basic-infixes ()
  "Prefix that just shows off many typical infix types."
  ["Infixes"

   ;; from macro
   ("-e" "exclusive switches" ts--exclusive-switches)

   ;; shorthand definitions
   ("-b" "switch with shortarg" ("-w" "--switch-short")) ; with :short-arg != :key
   ("-s" "switch" "--switch")
   ( "n" "no dash switch" "still works")
   ("-a" "argument" "--argument=" :prompt "Let's argue because: ")

   ;; a bit of inline EIEIO in our shorthand
   ("-n" "never empty" "--non-null=" :always-read t
    :init-value (lambda (obj) (oset obj value "better-than-nothing")))

   ("-c" "choices" "--choice=" :choices (foo bar baz))]

  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-basic-infixes)

Reading Infix Values

Reminder in the section on pre-commands the discussion about the :transient mentions that the values available in a suffix body depend on whe ther the pre-command called transient--export before evaluating the suffix body.

There are two basic ways to read infixes:

  • (transient-args transient-current-command) and parse manually
  • (transient-arg-value "--argument-" (transient-args transient-current-command)
  • (transient-suffixes transient-current-command) and retrieve your fully hydrated suffix

The transient-suffixes option requires filtering

In my opinion the API should make it easer to get raw values from suffixes, but this is also a matter of custom infixes needing to serialize values correctly so that transient-arg-value will “just work”.

Scope

When you call a function with an argument, you want to know in the body of your function what that argument was. This is the scope. The prefix is initialized with the :scope either in it’s own body or a similar form. Suffixes can then read back that scope in their body. The suffix object is given the scope and can use it to alter its own display or behavior. The layout also can interpret the scope while it is initializing.

WARNING When writing predicates against the scope, you will need to determine whether transient--prefix or transient-current-prefix is correct when writing prefix-generic suffixes. It is very subtle if you accidentally choose the wrong one and the parent has a nil scope while the child has an entirely different scope. These variables change throughout the lifecycle! Use edebug you must!

(transient-define-suffix ts--read-prefix-scope ()
  "Read the scope of the prefix."
  :transient 'transient--do-call
  (interactive)
  (let ((scope (oref transient-current-prefix scope)))
    (message "scope: %s" scope)))

(transient-define-suffix ts--double-scope-re-enter ()
  "Re-enter the current prefix with double the scope."
  ;; :transient 'transient--do-replace ; builds up the stack
  :transient 'transient--do-exit
  (interactive)
  (let ((scope (oref transient-current-prefix scope)))
    (if (numberp scope)
        (transient-setup transient-current-command nil nil :scope (* scope 2))
      (message (propertize (format "scope was non-numeric! %s" scope) 'face 'warning))
      (transient-setup transient-current-command))))

(transient-define-suffix ts--update-scope-with-prefix-re-enter (new-scope)
  "Re-enter the prefix with double the scope."
  ;; :transient 'transient--do-replace ; builds up the stack
  :transient 'transient--do-exit ; do not build up the stack
  (interactive "P")
  (message "universal arg: %s" new-scope)
  (transient-setup transient-current-command nil nil :scope new-scope))

(transient-define-prefix ts-scope (scope)
  "Prefix demonstrating use of scope."

  ;; note!  this is a location where we definitely had to use
  ;; `transient--prefix' or get the transient object from the ts-scope symbol.
  ;; `transient-current-prefix' is not correct here!
  [:description (lambda () (format "Scope: %s" (oref transient--prefix scope)))
   [("r" "read scope" ts--read-prefix-scope)
    ("d" "double scope" ts--double-scope-re-enter)
    ("o" "update scope (use prefix argument)" ts--update-scope-with-prefix-re-enter)]]
  (interactive "P")
  (transient-setup 'ts-scope nil nil :scope scope))

;; Setting an interactive argument for `eval-last-sexp' is a little different
;; (let ((current-prefix-arg 4)) (call-interactively 'ts-scope))

;; (ts-scope)
;; Then press "C-u 4 o" to update the scope
;; Then d to double
;; Then r to read
;; ... and so on
;; C-g to exit

Errata with prefix arg (C-u universal argument).

Key binding sequences, such as “wa” instead of single-key prefix bindings will unset the prefix argument (the old-school Emacs C-u prefix argument, not the prefix’s scope or other explicit arguments)

Possibly a bug in transient.

Prefix Value & History

Briefly, there are three locations for state you need to be aware of for this section:

  • Each transient’s prefix object has a :value that is updated by transient-set and transient-save
  • The values obtained from transient-args are usually quite ephemeral and don’t even persist beyond the body of form of the suffixes you usually read them in
  • transient-values contains saved values that are used to rehydrate the prefix :value slot when the prefix is created
  • transient-history is used to make it faster for the user to flip through previous states (which can have independent histories for infixes and prefixes). These are never used unless calling transient-history-prev and transient-history-next.

We can get this as a list of strings for any prefix by calling transient-args on transient-current-command in the suffix’s interactive form. If you know the command you want the value of, you can use it’s symbol instead of transient-current-command.

This is related to history keys. If you set the arguments and then save them using (C-x s) for the command transient-save, not only will the transient be updated with the new value, but if you call the child independently, it can still read the value from the suffix.

(transient-define-suffix ts-suffix-eat-snowcone (args)
  "Eat the snowcone!
This command can be called from it's parent, `ts-snowcone-eater' or independently."
  :transient t
  ;; you can use the interactive form of a command to obtain a default value
  ;; from the user etc if the one obtained from the parent is invalid.
  (interactive (list (transient-args 'ts-snowcone-eater)))

  ;; `transient-arg-value' can (with varying success) pick out individual
  ;; values from the results of `transient-args'.

  (let ((topping (transient-arg-value "--topping=" args))
        (flavor (transient-arg-value "--flavor=" args)))
    (message "I ate a %s flavored snowcone with %s on top!" flavor topping)))

(transient-define-prefix ts-snowcone-eater ()
  "Prefix demonstrating set & save infix persistence."

  ;; This prefix has a default value that ts-suffix-eat-snowcone can see
  ;; even before the prefix has been called.
  :value '("--topping=fruit" "--flavor=cherry")

  ;; always-read is used below so that you don't save nil values to history
  ["Arguments"
   ("-t" "topping" "--topping="
    :choices ("ice cream" "fruit" "whipped cream" "mochi")
    :always-read t)
   ("-f" "flavor" "--flavor="
    :choices ("grape" "orange" "cherry" "lime")
    :always-read t)]

  ;; Definitely check out the =C-x= menu
  ["C-x Menu Behaviors"
   ("S" "save snowcone settings"
    (lambda () (interactive) (message "saved!") (transient-save)) :transient t)
   ("R" "reset snowcone settings"
    (lambda () (interactive) (message "reset!") (transient-reset)) :transient t)]

  ["Actions"
   ("m" "message arguments" ts-suffix-print-args)
   ("e" "eat snowcone" ts-suffix-eat-snowcone)])

;; First call will use the transient's default value
;; M-x ts-suffix-eat-snowcone or `eval-last-sexp' below
;; (call-interactively 'ts-suffix-eat-snowcone)
;; (ts-snowcone-eater)
;; Eat some snowcones with different flavors
;; ...
;; ...
;; ...
;; Now save the value and exit the transient.
;; When you call the suffix independently, it can still read the saved values!
;; M-x ts-suffix-eat-snowcone or `eval-last-sexp' below
;; (call-interactively 'ts-suffix-eat-snowcone)

It’s worth bringing up the =transient-show-common-commands= variable. You may want to set this when working on the history support for your transients. Otherwise, just remember the (C-x) menu inside transients.

History Keys

History lets you set infixes using prior values. It’s per-prefix, per-suffix usually. Using previous examples like ts-snowcone-eater, you can flip through history using:

  • C-x p for transient-history-prev
  • C-x n for transient-history-next

These bindings are revealed when transient-show-common-commands is t or when you hit the C-x prefix.

However, what if you don’t want a unique history for some infixes or even prefixes?

Note As a more advanced example, using EIEIO and dynamic layout techniques to modify the slot of :history-key, you can also make unique histories for the same prefix/infix by setting that slot value depending on the context you want unique histories for.

The following example can demonstrate the behavior with some user effort:

(transient-define-prefix ts-ping ()
  "Prefix demonstrating history sharing."

  :history-key 'non-unique-name

  ["Ping"
   ("-g" "game" "--game=")
   ("p" "ping the pong" ts-pong)
   ("a" "print args" ts-suffix-print-args :transient nil)])

(transient-define-prefix ts-pong ()
  "Prefix demonstrating history sharing."

  :history-key 'non-unique-name

  ["Pong"
   ("-g" "game" "--game=")
   ("p" "pong the ping" ts-ping)
   ("a" "print args" ts-suffix-print-args :transient nil)])

;; (ts-ping)
;; Okay here's where it gets weird
;; 1.  Set the value of game to something and remember it
;; 2.  Press a to print the args
;; 3.  Re-open ts-ping.
;; 4.  C-x p to load the previous history, see the old value?
;; 5.  p to switch to the ts-pong transient
;; 6.  C-x p to load the previous history, see the old value from ts-ping???
;; 7. Note that ts-pong uses the same history as ts-ping!

Detangling with Initialization, Setting, and Saving

Set values show up in the prefix’s value slot.

(oref (plist-get (symbol-plist 'ts-ping) 'transient--prefix) value)

The prefix value will get the last value that was set using transient-set.

However, the prefix value shown in transient-values is only updated when calling transient-save.

Saved values show up in transient-values. If you save ts-ping, you can see the saved value here:

(assoc 'ts-ping transient-values)

These two values may be independent. They are written at the same time when calling transient-save. During prefix initializaton, the :value is written from transient-values.

Play with the ts-snowcone-eater and ts-ping and ts-pong in the C-x menu while also looking at what gets stored in transient-values, transient-history and the prefix’s slots.

When you re-evaluate the prefix or reload Emacs, you will see the result of initialization from transient-values.

Disabling Set / Save on a Suffix

To disable saving and setting values, causing a prefix to always end up using the default value, set the :unsavable slot to t.

(transient-define-prefix ts-goldfish ()
  "A prefix that cannot remember anything."
  ["Goldfish"
   ("-r" "rememeber" "--i-remember="
    :unsavable t ; infix isn't saved
    :always-read t ; infix always asks for new value
    ;; overriding the method to provide a starting value
    :init-value (lambda (obj) (oset obj value "nothing")))
   ("a" "print args" ts-suffix-print-args :transient nil)])

;; (ts-goldfish)

Try to update remember and then set and save it in the C-x menu. Reload it. It will never pay attention to history or setting & saving the transient value.

Setting or Saving Every Time a Suffix is Used

(transient-define-suffix ts-suffix-remember-and-wave ()
  "Wave, and force the prefix to set it's saveable infix values."
  (interactive)

  ;; (transient-reset) ; forget
  (transient-set) ; save for this session
  ;; If you combine reset with set, you get a reset for future sessions only.
  ;; (transient-save) ; save for this and future sessions
  ;; (transient-reset-value some-other-prefix-object)

  (message "Waves at user at: %s.  You will never be forgotten." (current-time-string)))

(transient-define-prefix ts-elephant ()
  "A prefix that always remembers its infixes."
  ["Elephant"
   ("-r" "rememeber" "--i-remember="
    :always-read t)
   ("w" "remember and wave" ts-suffix-remember-and-wave)
   ("a" "print args (skips remembering)" ts-suffix-print-args :transient nil)])

;; (ts-elephant)

Sticky infix support

There needs to be a slot that causes infixes to always be set on export. This would cover cases where the most frequent user input changes just rapidly enough that both setting every time and saving are equally inconvenient. Using transient-set is kind of brute-ish.

Default Values

Every transient prefix has a value. It’s a list. You can set it to create defaults for switches and arguments.

(transient-define-prefix ts-default-values ()
  "A prefix with a default value."

  :value '("--toggle" "--value=5")

  ["Arguments"
   ("t" "toggle" "--toggle")
   ("v" "value" "--value=" :prompt "an integer: ")]

  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-default-values)

Note, after setting or saving a value on this transient using the C-x menu, the next time the transient is set up, it will have a different value. If you want the default to return, use transient-reset in your suffix.

Readers

Readers are the mechanism to provide completions and to enforce input validity of infixes.

(transient-define-prefix ts-enforcing-inputs ()
  "A prefix with enforced input type."

  ["Arguments"
   ("v" "value" "--value=" :prompt "an integer: " :reader transient-read-number-N+)]

  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-enforcing-inputs)

Setting the reader can be used to enforce rules of valid input. See Advanced/Custom Infix Types for an example of writing a custom reader that validates input and assigning that reader via the class method instead of the :reader slot.

Lisp Variables

Lisp variables are currently at an experimental support level. They way they work is to report and set the value of a lisp symbol variable. Because they aren’t necessarilly intended to be printed as crude CLI arguments, they DO NOT appear in (transient-args 'prefix) but this is fine because you can just use the variable.

Customizing this class can be useful when working with objects and functions that exist entirely in elisp.

(defvar ts--position '(0 0) "A transient prefix location")

  (transient-define-infix ts--pos-infix ()
    "A location, key, or command symbol"
    :class 'transient-lisp-variable
    :transient t
    :prompt "An expression such as (0 0), \"p\", nil, 'ts--msg-pos: "
    :variable 'ts--position)

  (transient-define-suffix ts--msg-pos ()
    "Message the element at location"
    :transient 'transient--do-call
    (interactive)
    ;; lisp variables are not sent in the usual (transient-args) list.
    ;; Just read `ts--position' directly.
    (let ((suffix (transient-get-suffix transient-current-command ts--position)))
      (message "%s" (oref suffix description))))

  (transient-define-prefix ts-lisp-variable ()
    "A prefix that updates and uses a lisp variable."
    ["Location Printing"
     [("p" "position" ts--pos-infix)]
     [("m" "message" ts--msg-pos)]])

  ;; (ts-lisp-variable)

Controlling CLI’s

This section covers more usages of infixes, focused on creating better argument strings for CLI tools.

The section on flow control & managing state has more information about controlling elisp applications.

Reading arguments within suffixes

Note: these forms are generic for different prefixes, allowing you to mix and match suffixes within prefixes.

Switches & Arguments Again

The shorthand forms in transient-define-prefix are heavily influenced by the CLI style switches and arguments that transient was built to control. Most shorthand forms look like so:

("key" "description" "argument")

The macro will select the infix’s exact class depending on how you write :argument. If you write something ending in = such as --value= then you get :class transient-option but if not, the default is a :class transient-switch

Use =(describe-function transient-option)= and =(describe-function transient-option)= to see a full document of their slots and methods.

If you need an argument with a space instead of the equal sign, use a space and force the infix to be an argument by setting :class transient-option.

(transient-define-prefix ts-switches-and-arguments (arg)
  "A prefix with switch and argument examples."
  [["Arguments"
    ("-s" "switch" "--switch")
    ("-a" "argument" "--argument=")
    ("t" "toggle" "--toggle")
    ("v" "value" "--value=")]

   ["More Arguments"
    ("-f" "argument with forced class" "--forced-class " :class transient-option)
    ("I" "argument with inline" ("-i" "--inline-shortarg="))
    ("S" "inline shortarg switch" ("-n" "--inline-shortarg-switch"))]]

  ["Commands"
   ("w" "wave some" ts-wave)
   ("s" "show arguments" ts-suffix-print-args)]) ; use to analyze the switch values

;; (ts-switches-and-arguments)

Argument and Infix Macros

If you need to fine-tune a switch (boolean infix), use transient-define-infix. Likewise, use transient-define-argument for fine-tuning an argument. The class definitions can be used as a reference while the manual provides more explanation.

(transient-define-infix ts--random-init-infix ()
  "Switch on and off"
  :argument "--switch"
  :shortarg "-s" ; will be used for :key when key is not set
  :description "switch"
  :init-value (lambda (obj)
                (oset obj value
                      (eq 0 (random 2))))) ; write t with 50% probability

(transient-define-prefix ts-maybe-on ()
  "A prefix with a randomly intializing switch."
  ["Arguments"
   (ts--random-init-infix)]
  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-maybe-on)
;; (ts-maybe-on)
;; ...
;; Run the command a few times to see the random initialization of `ts--random-init-infix'
;; It will only take more than ten tries for one in a thousand users.  Good luck.

Choices

Choices can be set for an argument. The property API and transient-define-argument are equivalent for configuring choices. You can either hardcode or generate choices.

(transient-define-argument ts--animals-argument ()
  "Animal picker"
  :argument "--animals="
  ; :multi-value t ; multi-value can be set to --animals=fox,otter,kitten etc
  :class 'transient-option
  :choices '("fox" "kitten" "peregrine" "otter"))

(transient-define-prefix ts-animal-choices ()
  "Prefix demonstrating selecting animals from choices."
  ["Arguments"
   ("-a" "--animals=" ts--animals-argument)]
  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-animal-choices)

Choices shorthand in prefix definition

Choices can also be defined in a shorthand form. Use :class 'transient-option if you need to force a different class to be used.

(transient-define-prefix ts-animal-choices-shorthand ()
  "Prefix demonstrating the shorthand style of defining choices."
  ["Arguments"
   ("-a" "Animal" "--animal=" :choices ("fox" "kitten" "peregrine" "otter"))]
  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-animal-choices-shorthand)

Mutually Exclusive Switches

An argument with :class transient-switches may be used if a set of switches is exclusive. The key will likely not match the short argument. Regex is used to tell the interface that you are entering one of the choices. The selected choice will be inserted into :argument-format. The :argument-regexp must be able to match any of the valid options.

The UX on mutually exclusive switches is a bit of a pain to discover. You must repeatedly press =:key= in order to cycle through the options.

(transient-define-argument ts--snowcone-flavor ()
  :description "Flavor of snowcone"
  :class 'transient-switches
  :key "f"
  :argument-format "--%s-snowcone"
  :argument-regexp "\\(--\\(grape\\|orange\\|cherry\\|lime\\)-snowcone\\)"
  :choices '("grape" "orange" "cherry" "lime"))

(transient-define-prefix ts-exclusive-switches ()
  "Prefix demonstrating exclusive switches."
  :value '("--orange-snowcone")

  ["Arguments"
   (ts--snowcone-flavor)]
  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-exclusive-switches)

Incompatible Switches

If you need to prevent arguments in a group from being set simultaneously, you can set the prefix property :incompatible and a list of the long-style argument.

Use a list of lists, where each sublist is the long argument style. Match the string completely, including use of = in both arguments and switches.

(transient-define-prefix ts-incompatible ()
  "Prefix demonstrating incompatible switches."
  ;; update your transient version if you experience #129 / #155
  :incompatible '(("--switch" "--value=")
                  ("--switch" "--toggle" "--flip")
                  ("--argument=" "--value=" "--special-arg="))

  ["Arguments"
   ("-s" "switch" "--switch")
   ("-t" "toggle" "--toggle")
   ("-f" "flip" "--flip")

   ("-a" "argument" "--argument=")
   ("v" "value" "--value=")
   ("C-a" "special arg" "--special-arg=")]

  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-incompatible)

Short Args

This section is incomplete. Maybe Magit contains better answers.

Sometimes the :shortarg in a CLI doesn’t exactly match the :key: and :argument, so it can be specified manually.

The :shortarg concept could be used to help use man-pages or only for transient-detect-key-conflicts but it’s not clear what behavior it changes.

Shortarg cannot be used for exclusion excluding other options (prefix :incompatible) or setting default values (prefix :value).

Choices from a function

See transient-infix-read for actual code. This method uses the prefix’s history and then delecates to completing-read or completing-read-multiple. The :choices key coresponds to the COLLECTION argument passed to completing reads.

Note, using a function for completions can appear to require a daunting amount of behavior if you read the manul <a href=”info:elisp#Programmed Completion”>section on programmed completions. If you however just return a list of options, even when FLAG is not t, everything seems just fine.

(defun ts--animal-choices (complete-me predicate flag)
  ;; complete-me: whatever the user has typed so far
  ;; predicate: function you should use to filter candidates (only nil seen so far)
  ;; flag: request for metadata (which can be disrespected)

  ;; if you want to respect metadata requests, here's what the form might
  ;; look like, but no behavior was observed.
  (if (eq flag 'metadata)
      '(metadata . '((annotation-function . (lambda (c) "an annotation"))))

    ;; when not handling a metadata request from completions, use some
    ;; logic to generate the choices, possibly based on input or some time
    ;; / context sensitive process.  FLAG will be `t' when these are reqeusted.
    (if (eq 0 (random 2))
        '("fox" "kitten" "otter")
      '("ant" "peregrine" "zebra"))))

(transient-define-prefix ts-choices-with-completions ()
  "Prefix with completions for choices."
  ["Arguments"
   ("-a" "Animal" "--animal="
    :always-read t ; don't allow unsetting, just read a new value
    :choices ts--animal-choices)]
  ["Show Args"
   ("s" "show arguments" ts-suffix-print-args)])

;; (ts-choices-with-completions)

multiple instances

Switches and arguments that can be used multiple times are supported. Example needs to be written. This is useful for CLI wrapping or perhaps situations where a command accepts multiple levels of the same setting.

Dispatching args into a process

If you want to call a command line application using the arguments, you might need to do a bit of work processing the arguments. The following example uses cowsay.

  • Cowsay doesn’t actually have a message= argument, So we end up stripping it from the arguments and re-assembling something call-process can use.
  • Cowsay supports more options, but for the sake of keeping this example small (and to refocus effort on transient itself), the set of all CLI options are not fully supported.

There’s some errata about this example:

  • The predicates don’t update the transient. (transient--redisplay) doesn’t do the trick. We could use transient--do-replace and transient-setup, but that would lose existing state
  • The predicate needs to be exists & not empty (but doesn’t matter yet)
(defun ts--quit-cowsay ()
  "Kill the cowsay buffer and exit"
  (interactive)
  (kill-buffer "*cowsay*"))

(defun ts--cowsay-buffer-exists-p ()
  (not (equal (get-buffer "*cowsay*") nil)))

(transient-define-suffix ts--cowsay-clear-buffer (&optional buffer)
  "Delete the *cowsay* buffer.  Optional BUFFER name."
  :transient 'transient--do-call
  :if 'ts--cowsay-buffer-exists-p
  (interactive) ; todo look at "b" interactive code

  (save-excursion
    (let ((buffer (or buffer "*cowsay*")))
      (set-buffer buffer)
      (delete-region 1 (+ 1 (buffer-size))))))

(transient-define-suffix ts--cowsay (&optional args)
  "Run cowsay"
  (interactive (list (transient-args transient-current-command)))
  (let* ((buffer "*cowsay*")
         ;; TODO ugly
         (cowmsg (if args (transient-arg-value "--message=" args) nil))
         (cowmsg (if cowmsg (list cowmsg) nil))
         (args (if args
                   (seq-filter
                    (lambda (s) (not (string-prefix-p "--message=" s))) args)
                 nil))
         (args (if args
                   (if cowmsg
                       (append args cowmsg)
                     args)
                 cowmsg)))

    (when (ts--cowsay-buffer-exists-p)
      (ts--cowsay-clear-buffer))
    (apply #'call-process "cowsay" nil buffer nil args)
    (switch-to-buffer buffer)))

(transient-define-prefix ts-cowsay ()
  "Say things with animals!"

  ; only one kind of eyes is meaningful at a time
  :incompatible '(("-b" "-g" "-p" "-s" "-t" "-w" "-y"))

  ["Message"
   ("m" "message" "--message=" :always-read t)] ; always-read, so clear by entering empty string
  [["Built-in Eyes"
    ("b" "borg" "-b")
    ("g" "greedy" "-g")
    ("p" "paranoid" "-p")
    ("s" "stoned" "-s")
    ("t" "tired" "-t")
    ("w" "wired" "-w")
    ("y" "youthful" "-y")]
   ["Actions"
    ("c" "cowsay" ts--cowsay :transient transient--do-call)
    ""
    ("d" "delete buffer" ts--cowsay-clear-buffer)
    ("q" "quit" ts--quit-cowsay)]])

;; (ts-cowsay)

Cleanup Cowsay

Clean up cowsay example. Check for binary before attempting to run it.

Controlling Visibility

At times, you need a prefix to show or hide certain options depending on the context.

Visibility Predicates

Simple predicates at the group or element level exist to hide parts of the transient when they wouldn’t be useful at all in the situation.

(defvar ts-busy nil "Are we busy?")

(defun ts--busy-p () "Are we busy?" ts-busy)

(transient-define-suffix ts--toggle-busy ()
  "Toggle busy"
  (interactive)
  (setf ts-busy (not ts-busy))
  (message (propertize (format "busy: %s" ts-busy)
                       'face 'success)))

Open the following example in buffers with different modes (or change modes manually) to see the different effects of the mode predicates.

(transient-define-prefix ts-visibility-predicates ()
  "Prefix with visibility predicates.
Try opening this prefix in buffers with modes deriving from different
abstract major modes."
  ["Empty Groups Not Displayed"
   ;; in org mode for example, this group doesn't appear.
   ("we" "wave elisp" ts-suffix-wave :if-mode emacs-lisp-mode)
   ("wc" "wave in C" ts-suffix-wave :if-mode cc-mode)]

  ["Lists of Modes"
   ("wm" "wave multiply" ts-suffix-wave :if-mode (dired-mode gnus-mode))]

  [["Function Predicates"
    ;; note, after toggling, the transient needs to be re-displayed for the
    ;; predicate to take effect
    ("b" "toggle busy" ts--toggle-busy)
    ("bw" "wave busily" ts-suffix-wave :if ts--busy-p)]

   ["Programming Actions"
    :if-derived prog-mode
    ("pw" "wave programishly" ts-suffix-wave)
    ("pe" "wave in elisp" ts-suffix-wave :if emacs-lisp-mode)]
   ["Special Mode Actions"
    :if-derived special-mode
    ("sw" "wave specially" ts-suffix-wave)
    ("sd" "wave dired" ts-suffix-wave :if-mode dired-mode)]
   ["Text Mode Actions"
    :if-derived text-mode
    ("tw" "wave textually" ts-suffix-wave)
    ("to" "wave org-modeishly" ts-suffix-wave :if-mode org-mode)]])

;; (ts-visibility-predicates)

Inapt (Temporarily Unavailable)

“Greyed out” suffixes. Inapt is better if an option is temporarily unavailable due to a state that varies with each invocation of the transient.

Inapt predicates work on suffixes, but not on groups (which would have to modify every child).

Note, like visibility predicates, inapt-* predicates do not take effect until the transient has it’s layout fully redone. Therefore this example uses a child transient and updates the scope.

(defun ts--child-scope-p ()
  "Returns the scope of the current transient.
When this is called in layouts, it's the transient being layed out"
  (let ((scope (oref transient--prefix scope)))
    (message "The scope is: %s" scope)
    scope))

;; the wave suffixes were :transient t as defined, so we need to manually
;; override them to the `transient--do-return' value for :transient slot so
;; that they return back to the parent.
(transient-define-prefix ts--inapt-children ()
  "Prefix with children using inapt predicates."
  ["Inapt Predicates Child"
   ("s" "switched" ts--wave-surely
    :transient transient--do-return
    :if ts--child-scope-p)
   ("u" "unswitched" ts--wave-normally
    :transient transient--do-return
    :if-not ts--child-scope-p)]

  ;; in the body, we read the value of the parent and set our scope to
  ;; non-nil if the switch is set
  (interactive)
  (let ((scope (transient-arg-value "--switch" (transient-args 'ts-inapt-parent))))
    (message "scope: %s" scope)
    (message "type: %s" (type-of scope))
    (transient-setup 'ts--inapt-children nil nil :scope (if scope t nil))))

(transient-define-prefix ts-inapt-parent ()
  "Prefix that configures child with inapt predicates."

  [("-s" "switch" "--switch")
   ("a" "show arguments" ts-suffix-print-args)
   ("c" "launch child prefix" ts--inapt-children :transient transient--do-recurse)])

;; (ts-inapt-parent)

Documentation in manual missing

There is not a single mention of inapt even though it’s fully implemented and works.

Levels

Levels are another way to control visibility.

  • As a developer, you set levels to optionally expose or hide children in a prefix.
  • As a user, you change the prefix’s level and the levels of suffixes to customize what’s visible in the transient.

Lower levels are more visible. Setting the level higher reveals more suffixes. 1-7 are valid levels.

The user can adjust levels within a transient prefix by using (C-x l) for transient-set-level. The default active level is 4, stored in transient-default-level. The default level for children is 1, stored in transient--default-child-level.

Per-suffix and per-group, the user can set the level at which the child will be visible. Each prefix has an active level, remembered per prefix. If the child level is less-than-or-equal to the child level, the child is visible.

A hidden group will hide a suffix even if that suffix is at a low enough level. Issue #153 has some addional information about behavior that might get cleaned up.

Defining group & suffix levels

Adding default levels for children is as simple as adding integers at the beginning of each list or vector. If some commands are not likely to be used, instead of making the hard choice to include them or not, you can provide them, but tell the user in your README to set higher levels.

(transient-define-prefix ts-levels-and-visibility ()
  "Prefix with visibility levels for hiding rarely used commands."

  [["Setting the Current Level"
    ;; this binding is normally not displayed.  The value of
    ;; `transient-show-common-commands' controls this by default.
    ("C-x l" "set level" transient-set-level)
    ("s" "show level" ts-suffix-show-level)]

   [2 "Per Group" ; 1 is the default default-child-level
      ("ws" "wave surely" ts--wave-surely) ; 1 is the default default-child-level
      (3"wn" "wave normally" ts--wave-normally)
      (5"wb" "wave non-essentially" ts--wave-non-essentially)]

   [3 "Per Group Somewhat Useful"
      ("wd" "wave definitely" ts--wave-definitely)]

   [6 "Groups hide visible children"
      (1 "wh" "wave hidden" ts--wave-hidden)]

   [5 "Per Group Rarely Useful"
      ("we" "wave eventually" ts--wave-eventually)]])

;; (ts-levels-and-visibility)

Using the Levels UI

Press (C-x l) to open the levels UI for the user. Press (C-x l) again to change the active level. Press a key such as “we” to change the level for a child. After you cancel level editing with (C-g), you will see that children have either become visible or invisible depending on the changes you made.

*While a child may be visible according to its own level, if it’s hidden within the group, the user’s level-setting UI for the prefix will contradict what’s actually visible. The UI does not allow setting group levels.*

Advanced

The previous sections are designed to go breadth-first so that you can get core ideas first. The following examples expand on combinations of several ideas or subclassing & customizing rarely used slots.

Some of these examples are approaching the complexity of just reading magit source.

Dynamically generating layouts

While you can cover many cases using predicates, layouts, and visibility, sometimes you really do want to generate a list of commands.

Note, beware that you could be creating a lot of suffix objects if the forms you use generate unique symbols. These will pollute command completions over time, so probably don’t do that.

transient-setup-children

This is a group method that can be overridden in order to modify or eliminate some children from display. If you need a central place for children to coordinate some behavior, this may work for you.

(transient-define-prefix ts-generated-child ()
  "Prefix that uses `setup-children' to generate single child."

  ["Replace this child"
   ;; Let's override the group's method
   :setup-children
   (lambda (_) ; we don't care about the stupid suffix

     ;; remember to return a list
     (list (transient-parse-suffix
            transient--prefix
            '("r" "replacement" (lambda ()
                                  (interactive)
                                  (message "okay!"))))))

   ("s" "haha stupid suffix" (lambda ()
                               (interactive)
                               (message "You should replace me!")))])

;; (ts-generated-child)

transient--parse-child takes the same configuration format as transient-define-prefix. You can see the layout format in the layout hacking appendix. transient--prarse-group works almost exactly the same, just for groups.

The same thing, but parsing an entire group spec:

(transient-define-prefix ts-generated-group ()
  "Prefix that uses `setup-children' to generate a group."

  ["Replace this child"
   ;; Let's override the group's method
   :setup-children
   (lambda (_) ; we don't care about the stupid suffix

     ;; the result of parsing here will be a group
     (transient-parse-suffixes
      transient--prefix
      ["Group Name" ("r" "replacement" (lambda ()
                                         (interactive)
                                         (message "okay!")))]))

   ("s" "haha stupid suffix" (lambda ()
                               (interactive)
                               (message "You should replace me!")))])

;; (ts-generated-group)

If you need to define a dynamic group class, override transient-setup-children. It will work almost entirely the same as the examples above. Set your group class in the prefix using the :class key.

Note you don’t need to be inside of a layout body to hack around with dynamic layouts. Mess around in ielm.

(transient--parse-child 'magit-dispatch '("a" "action" (lambda () (interactive) (message "hey"))))

Note you can replace transient--prefix with ts-generated-group in the example above. transient--prefix is just a variable that happens to hold the prefix during layout.

Correction in manual

  • These functions do mostly the same job. Why do we need to specify a prefix for transient-parse-suffixes, for scope etc?

Using prefix scope in children

Basically you are on your own. Just call (oref transient--prefix scope) during layout setup or (oref transient-current-prefix) during suffix bodies.

Obtaining Missing Scope

Because suffixes are basically also commands (riding in the same symbol plist), a suffix can be called independently. In this case, if its expecting to read the scope from the prefix when there is no prefix, it might fail.

Therefore, a method called transient-init-scope can be overridden and used at the correct point in the lifecycle for the suffix to correct the issue.

Note, the behavior is actually quite ad-hoc. You will read the prefix yourself and then decide if you want to use some fallback.

There is a perfectly short example in Magit source for the custom magit--git-variable subclass of the transient-variable infix.

Each infix instance is declared in transient-define-infix, potentially with a :scope slot, possibly holding a function.

If it’s holding a function, that function will be used as a backup during initialization in case there is no prefix or it has nothing in its scope slot.

Custom Infix Types

Not everything is a string or boolean. You may want to represent complex objects in your transint infixes. If your objects can be rehydrated from some serialized ID, you may want history support.

If you need to set and display a custom type, use the simple OOP techniques of EIEIO. Also check the suffix value methods section of the transient manual.

Essential behaviors for your custom infix:

  • Defining a reader to set the infix with user input
  • prompt slot’s default form, initform for asking the user for input
  • transient-init-value to rehydrate saved values
  • transient-infix-value so that setting & saving persist what you want to rehydrate
  • transient-format-value to display a user-meaningful form for your value

We will also use some layout introspection:

  • transient-get-suffix To get suffix by key, location, or command symbol
  • Getting a description from raw layout children (not EIEIO objects). See Layout Hacking.
;; The children we will be picking can be of several forms.  The
;; transient--layout symbol property of a prefix is a vector of vectors, lists,
;; and strings.  It's not the actual eieio types or we would use
;; `transient-format-description' to just ask them for the descriptions.
(defun ts--layout-child-desc (layout-child)
  "Get the description from a transient layout vector or list."
  (let ((description
         (cond
          ((vectorp layout-child) (or (plist-get (aref layout-child 2) :description) "<group, no desc>")) ; group
          ((stringp layout-child) layout-child) ; plain-text child
          ((listp layout-child) (plist-get (elt layout-child 2) :description)) ; suffix
          (t (message (propertize "You traversed into a child's list elements!" 'face 'warning))
             (format "(child's interior) element: %s" layout-child)))))
    (cond
     ;; The description is sometimes a callable function with no arguments,
     ;; so let's call it in that case.  Note, the description may be
     ;; designed for one point in the transient's lifecycle but we could
     ;; call it in a different one, causing its behavior to change.
     ((functionp description) (apply description))
     (t description))))

;; We repeat the read using a lisp expression from `read-from-minibuffer' to get
;; the LOC key for `transient-get-suffix' until we get a valid result.  This
;; ensures we don't store an invalid LOC.
(defun ts-child-infix--reader (prompt initial-input history)
  "Read a location and check that it exists within the current transient."
  (let ((command (oref transient--prefix command))
        (success nil))
    (while (not success)
      (let* ((loc (read (read-from-minibuffer prompt initial-input nil nil history)))
             (child (ignore-errors (transient-get-suffix command loc))))
        (if child (setq success loc)
          (message (propertize
            (format
             "Location could not be found in prefix %s"
             command) 'face 'error)) (sit-for 3))))
    success))

;; Inherit from variable abstract class
(defclass ts-child-infix (transient-variable)
  ((value-object :initarg value-object :initform nil)
   ;; this is a new slot for storing the hydrated value.  we re-use the
   ;; value infrastructure for storing the serialization-friendly value,
   ;; which is basically a suffix addres or id.

   (reader :initform #'ts-child-infix--reader)
   (prompt :initform "Location, a key \"c\", suffix-command-symbol like ts--wave-normally or coordinates like (0 2 0): ")))

;; We have to define this on non-abstract infix classes.  See
;; `transient-init-value' in transient source.  The method on
;; `transient-argument' class is the best example for initializing your
;; suffix based on the prefix's value, but it does support a lot of
;; behaviors.
(cl-defmethod transient-init-value ((obj ts-child-infix))
  "Set the value and object-value using the prefix's value."
  (let* ((prefix-value (oref transient--prefix value))
         (key (oref obj command))
         (value (car (alist-get key prefix-value))) ; car?
         (value-object (transient-get-suffix (oref transient--prefix command) value)))
    (oset obj value value)
    (oset obj value-object value-object)))

(cl-defmethod transient-infix-set ((obj ts-child-infix) value)
  "When the `value' is updated, update the `value-object' as well."
  (let* ((command (oref transient--prefix command))
         (child (ignore-errors (transient-get-suffix command value))))
    (oset obj value-object child)
    (oset obj value (if child value nil))))

;; If you are making a suffix that needs history, you need to define this
;; method.  You also need this method if your value needs some processing
;; or use of an alternate value for later rehydration.  Tell the prefix
;; what to store when setting / saving
(cl-defmethod transient-infix-value ((obj ts-child-infix))
  "Return our actual value for rehydration later."

  ;; this is almost identical to the method defined for `transient-infix',
  ;; but don't forget this if you want history on a suffix for example.
  (list (oref obj command) (oref obj value)))

;; Show user's a useful representation of your ugly value
(cl-defmethod transient-format-value ((obj ts-child-infix))
  "All transient children have some description we can display.
Show either the child's description or a default if no child is selected."
  (if-let* ((value (and (slot-boundp obj 'value) (oref obj value)))
            (value-object (and (slot-boundp obj 'value-object)
                               (oref obj value-object))))
      (propertize
       (format "(%s)" (ts--layout-child-desc value-object))
       'face 'transient-value)
    (propertize "¯\_(ツ)_/¯" 'face 'transient-inactive-value)))

;; Now that we have our class defined, we can create an infix the usual
;; way, just specifying our class
(transient-define-infix ts--inception-child-infix ()
  :class ts-child-infix)

;; All set!  This transient just tests our or new toy.
(transient-define-prefix ts-inception ()
  "Prefix that picks a suffix from its own layout."

  [["Pick a suffix"
    ("-s" "just a switch" "--switch") ; makes history value structure apparent
    ("c" "child" ts--inception-child-infix :class ts-child-infix)]

   ["Some suffixes"
    ("s" "wave surely" ts--wave-surely)
    ("d" "wave definitely" ts--wave-definitely)
    ("e" "wave eventually" ts--wave-eventually)
    ("C" "call & exit normally" ts--wave-normally :transient nil)]

   ["Read variables"
    ("r" "read args" ts-suffix-print-args )]])

;; (ts-inception)
;; Try setting the infix to "e" (yes, include quotes)
;; Try: (1 2)
;; Try: ts--wave-normally
;; Set the infix and re-open it
;; Save the infix, re-evaluate the prefix, and open the prefix again
;; Try flipping through history
;; Now do think of doing things like this with org ids, magit-sections, buffers etc.

This is a difficult example, but once you understand the pieces, you can see some of the magit variables in action like magit--git-variable and it’s many subclasses.

Revisit the section on detangling setting, saving and history. Watching the values update will make it clear what representations are bing stored, where, and when.

Reading custom infix values

Note, however you store and rehydrate will affect how you read, so try to make it just work with transient-read-arg, unlike this example (TODO).

 (transient-define-suffix ts--inception-update-description ()
   "Update the description of of the selected child."
   (interactive)
   (let* ((args (transient-args transient-current-command))
          (description (transient-arg-value "--description=" args))
          ;; This is the part where we read the other infix
          (loc (car (cdr (assoc 'ts--inception-child-infix args))))
          (layout-child (transient-get-suffix 'ts-inception-update loc)))
     (cond
      ;; Once again, do different bodies based on what we found at the layout locition.
      ((or (listp layout-child) ; child
          (vectorp layout-child) ; group
          (stringp layout-child)) ; string child
       (if (stringp layout-child)
           (transient-replace-suffix 'ts-inception-update loc description) ; plain-text child
         (plist-put (elt layout-child 2) :description description)))
      (t (message (propertize (format
                               "Don't know how to modify whatever is at: %s"
                               loc) 'face 'warning))))
     ;; re-enter the transient manually to display the modified layout
     (transient-setup transient-current-command)))

(transient-define-prefix ts-inception-update ()
  "Prefix that picks and updates its own suffix."

  [["Pick a suffix"
    ("c" "child" ts--inception-child-infix)]

   ["Update the description!"
    ("-d" "description" "--description=") ; makes history value structure apparent
    ("u" "update" ts--inception-update-description :transient transient--do-exit)]

   ["Some suffixes"
    ("s" "wave surely" ts--wave-surely)
    ("d" "wave definitely" ts--wave-definitely)
    ("e" "wave eventually" ts--wave-eventually)
    ("C" "call & exit normally" ts--wave-normally :transient nil)]

   ["Read variables"
    ("r" "read args" ts-suffix-print-args )]])

;; (ts-inception-update)
;; Pick a suffix,
;; Then set the description
;; Then update the suffix's you picked with the new description!
;; Using a transient to modify a transient (⊃。•́‿•̀。)⊃━✿✿✿✿✿✿
;; Try to rename a group, such as (0 0)
;; Rename the very outer group, (0)

Errata

Modifying the very outer group doesn’t quite work. It’s probably a degenrate layout object, meaning setting a description doesn’t cause it to behave like a group with a heading. Maybe outer groups have a different data structure? An exercise left to the reader

The flow control for re-display is slightly fighting the history implementation. It would be better if we could retain values while triggering a redraw without even more hacking & state manipulation.

Appendixes

EIEIO - OOP in Elisp

Emacs lisp ships with eieio, a close cousin to the Common Lisp Object System. It’s OOP. There are classes & subclasses. You can inherit into new classes and override methods to customize behaviors.

You can use eieio API’s to explore transient objects. Let’s look at some transients you have already:

;; The plist for a prefix command contains a `transient-prefix' object in the
;; `transient--prefix' key and a vector layout in `transient--layout' (symbol-plist
(symbol-plist 'magit-dispatch)

;; getting the values from the symbol plist
(plist-get (symbol-plist 'magit-dispatch) 'transient--prefix)

(let ((prefix-object (plist-get (symbol-plist 'magit-dispatch) 'transient--prefix)))

  ;; printing the current slot values for that object
  (object-write prefix-object)

  ;; ;; Object transient-prefix-20997da
  ;; (transient-prefix "transient-prefix-20997da"
  ;;   :command magit-dispatch  :info-manual "(magit)Top")

  ;; getting the class of an object
  (eieio-object-class prefix-object) ; transient-prefix

  ;; opening the help documents for the class, which shows all methods and
  ;; slot forms
  (describe-function transient-prefix))

Typical OOP

Like all OOP, the three things you want to do are:

Override methods

cl-defmethod and sometimes cl-call-next-method

Override default values

Inside the defclass form, you can set slots that you don’t like. :initform is a default value. :initarg configures which argument to pick up from the class constructor.

Read & Update

oref and oset

Call Methods

(method-name object arguments)

Introspection

See methods like slot-boundp in the EIEIO eieio method index

Transient’s defclass’s and their inheritance

Here’s a list of all of transient’s defclass and their ancestry. This is how it is in 2022.

(eieio-browse) ; shows all known classes and their ancestry

;;     +--transient-child
;;     |    +--transient-group
;;     |    |    +--transient-subgroups
;;     |    |    +--transient-columns
;;     |    |    +--transient-row
;;     |    |    +--transient-column
;;     |    +--transient-suffix
;;     |         +--magit--git-submodule-suffix
;;     |         +--transient-infix
;;     |              +--transient-variable
;;     |              |    +--magit--git-variable
;;     |              |    |    +--magit--git-branch:upstream
;;     |              |    |    +--magit--git-variable:urls
;;     |              |    |    +--magit--git-variable:choices
;;     |              |    |         +--magit--git-variable:boolean
;;     |              |    +--transient-lisp-variable
;;     |              +--transient-argument
;;     |                   +--transient-switches
;;     |                   +--transient-option
;;     |                   |    +--transient-files
;;     |                   +--transient-switch
;;     +--transient-prefix
;;          +--magit-log-prefix
;;          |    +--magit-log-refresh-prefix
;;          +--magit-diff-prefix
;;               +--magit-diff-refresh-prefix

View Class Methods and Attributes

Using describe-function is extremely handly for viewing the class slots and methods.

Classes used in transient that you are likely to want to know the slots for:

transient-prefix transient-suffix transient-infix transient-argument

The eieio docs have a more wordy treatment. The class system has a lot of behavior that can be faster at times to just understand through description.

Debugging

There is a lot of support for both print-line and step-through debugging.

Print debug messages

Just set transient--debug to t. (setq transient-debug t)

You will get a lot of logs visible in *Messages* via view-echo-message-area the next time you run a transient.

-- setup              (cmd: ts-layout-rows-explicit, event: "M-x", exit: nil)
-- stack-zap          (cmd: ts-layout-rows-explicit, event: "M-x", exit: nil)
-- init-transient     (cmd: ts-layout-rows-explicit, event: "M-x", exit: nil)
     push transient--transient-map
     push transient--redisplay-map
-- post-command       (cmd: ts-layout-rows-explicit, event: "M-x", exit: nil)
-- pre-command        (cmd: transient-update, event: "w", exit: nil)
     pop  transient--redisplay-map
-- post-command       (cmd: transient-update, event: "w", exit: nil)
     pop  transient--redisplay-map
     push transient--redisplay-map
-- pre-command        (cmd: ts-suffix-wave, event: "w l", exit: nil)
-- stack-zap          (cmd: ts-suffix-wave, event: "w l", exit: nil)
-- pre-exit           (cmd: ts-suffix-wave, event: "w l", exit: t)
     pop  transient--transient-map
     pop  transient--redisplay-map
Waves at the user at: Sat Nov 12 22:38:20 2022.
-- post-command       (cmd: ts-suffix-wave, event: "w l", exit: t)
-- post-exit          (cmd: ts-suffix-wave, event: "w l", exit: t)

Watching evaluation in Edebug

Edebug works with transients. There is much support in transient to facilitate using edebug.

For watching the flow control around your command, especially helpful for debugging behavior around setup, layout, or suffix dispatch, you might want to watch your transient in Edebug.

Edebug basic introduction video (10 min).

In short:

  • goto your transient source
  • instrument a function you want to watch with edebug-defun
  • call the transient / suffix that triggers entry of that function
  • use SPC to step forward, c to continue, i to enter a function call, or h for help etc

First watch the debug output to gain an idea of how your code flows with the transient code. Then instrument transient behaviors such as transient--post-exit and use i to edebug-step-in to calls of interest.

When you are done, remember to use =edebug-remove-instrumentation= so that you can go on without every transient you open trying to call the debugger.

Debugging Macro Forms

Because edebug works on defuns while suffixes are defined with macros, you may need to macro exand in order to come up with something debuggable.

Layout Hacking

First you need to explort the layout data structures.

;; Let's look at the layout
(let ((prefix-layout (plist-get (symbol-plist 'magit-dispatch) 'transient--layout)))

  (type-of prefix-layout) ; cons

  (listp prefix-layout) ; t

  (length prefix-layout) ; 3

  ;; Each group in the list is a vector
  (vectorp (car prefix-layout)) ; t

  (elt (car prefix-layout) 0) ; first element is a priority
  (elt (car prefix-layout) 1) ; second is a type name
  (elt (car prefix-layout) 2) ; contents & attributes

  ;; the attributes are key-value pairs used to create the class
  ;; instance when the transient is shown.

  ;; the nested contents will be lists of vectors for groups and
  ;; lists of lists for suffixes.

  )

;; A sample layout

;; ([1 transient-column nil
;;     ((1 transient-suffix
;;         (:key "i" :description "Ignore" :command magit-gitignore))
;;      (1 transient-suffix
;;         (:key "I" :description "Init" :command magit-init))
;;      (1 transient-suffix
;;         (:key "j" :description "Jump to section" :command magit-status-jump :if-mode magit-status-mode))
;;      (1 transient-suffix
;;         (:key "j" :description "Display status" :command magit-status-quick :if-not-mode magit-status-mode)))])

  

You might find this helpful when constructing dynamic layouts

Hooks

Just a reminder, some hooks exist. Use describe-variable and complete with transient hook for the most recent list of hooks.

Preludes

Definitions that are not that interesting on their own but are used in examples.

ts-suffix-wave Command

(defun ts-suffix-wave ()
  "Wave at the user"
  (interactive)
  (message "Waves at the user at: %s." (current-time-string)))

ts-suffix-show-level

(transient-define-suffix ts-suffix-show-level ()
  "Show the current transient's level."
  :transient t
  (interactive)
  (message "Current level: %s" (oref transient-current-prefix level)))

ts–define-waver

;; Because command names are used to store and lookup child levels, we have
;; define a macro to generate unqiquely named wavers.  See #153 at
;; https://github.com/magit/transient/issues/153
(defmacro ts--define-waver (name)
  "Define a new suffix named ts--wave-NAME"
  `(transient-define-suffix ,(intern (format "ts--wave-%s" name)) ()
     ,(format "Wave at the user %s" name)
     :transient t
     (interactive)
     (message (format "Waves at %s" (current-time-string)))))

;; Each form results in a unique suffix definition.
(ts--define-waver "surely")
(ts--define-waver "normally")
(ts--define-waver "non-essentially")
(ts--define-waver "definitely")
(ts--define-waver "eventually")
(ts--define-waver "hidden")

ts-suffix-print-args

Here’s a suffix that reads the transient’s infix values, the prefix’s scope, and any universal argument (C-u 4 etc).

(transient-define-suffix ts-suffix-print-args (prefix-arg)
  "Report the universal argument, prefix's scope, and infix values."
  :transient 'transient--do-call
  (interactive "P")
  (let ((args (transient-args (oref transient-current-prefix command)))
        (scope (oref transient-current-prefix scope)))
    (message "prefix-arg: %s \nprefix's scope value: %s \ntransient-args: %s"
             prefix-arg scope args)))

Essential Elisp

If you were hit in the face with the first example, you need to learn basic Elisp. This is not an Elisp guide. Here’s some starting points:

  • transient-define-prefix is a macro that creates a command and attaches a transient-prefix object to the command symbol’s plist.
  • lambda is a macro to create an anonymous function
  • interactive is a macro that makes the function compatible with the command interface, the M-x or execute-extended-command menu
  • The brackets are just vector syntax.

    Besides the other ways to evaluate elisp used in this README, try ielm.

    Use the built-in elisp manual by calling the command elisp-index-search. See shortdocs for functions using shortdoc-display-groups.

    The EIEIO and CL manuals are independent from the Elisp manual for some reason. EIEIO is pretty short and not used much once you get the hang of it. info-display-manual

    Common Lisp manual you don’t really need the common lisp manual for working with transient. Don’t be alarmed when you see EIEIO using functions like cl-call-next-method

Further Reading

  • *The Transient Manual* (web link) contains more detailed explanation of behavior. The examples here should allow you to visualize what is being described. This guide and the manual should be your first and second sources.
  • *Transient source* (web link) is all in one file. Source code is always more accurate than manual descriptions, even if some behavior implementations are a bit scattered.
  • *Magit source* (web link) contains numerous examples of transient being used in a big, full-feature application. Search the source for “transient” and you will find many prefixes, suffixes, and custom classes. The smallest examples may be harder to find and most combine many behaviors at once.
<<package-footer>>

More Packaging

The headers and footers for the tangled module.

Package Header

;;; transient-showcase.el --- transient features & behavior showcase -*- lexical-binding: t; -*-

;; Copyright (C) 2022 Positron Solutions


;; Author: Psionik K <[email protected]>
;; Keywords: convenience
;; Version: 0.1.0
;; Package-Requires: ((emacs "28.1"))
;; Homepage: http://github.com/positron-solutions/transient-showcase

;;; License notice:

;; Permission is hereby granted, free of charge, to any person obtaining a
;; copy of this software and associated documentation files (the "Software"),
;; to deal in the Software without restriction, including without limitation
;; the rights to use, copy, modify, merge, publish, distribute, sublicense,
;; and/or sell copies of the Software, and to permit persons to whom the
;; Software is furnished to do so, subject to the following conditions:

;; The above copyright notice and this permission notice shall be included in
;; all copies or substantial portions of the Software.

;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
;; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
;; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
;; AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
;; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
;; FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
;; DEALINGS IN THE SOFTWARE.

;;; Commentary:

;; This package is created from the README and serves as a fast way to load
;; all of the examples without tangling the org document.  This is appropriate
;; if you just want to quickly browse through the examples and see their
;; source code.
;;
;; M-x transient-showcase contains most of the prefixes and can be bound for
;; use as a quick reference.  Just use transient's help for each command to
;; see the source. C-h <suffix key>.
;;

;;; Code:

(require 'transient)

Package Footer

This block includes the showcase transient.

(transient-define-prefix ts-showcase ()
  "A launcher for a currated selection of examples.
While most of the prefixes have their :transient slot set to t, it's not
possible to return from all of them, especially if they demonstrate flow
control such as replacing or exiting."

  [["Layouts"
    ("ls" "stacked" ts-layout-stacked :transient t)
    ("lc" "columns" ts-layout-columns :transient t)
    ("lt" "stacked columns" ts-layout-stacked-columns :transient t)
    ("lg" "grid" ts-layout-the-grid :transient t)
    ("lp" "spaced out" ts-layout-spaced-out :transient t)
    ("le" "explicit class" ts-layout-explicit-classes :transient t)
    ("ld" "descriptions" ts-layout-descriptions :transient t)
    ;; padded description to sc
    ("lD" "dynamic descriptions        " ts-layout-dynamic-descriptions :transient t)]

   ["Nesting & Flow Control"
    ("fs" "stay transient" ts-stay-transient :transient t)
    ("fb" "binding sub-prefix" ts-simple-parent :transient t)
    ("fr" "sub-prefix with return" ts-simple-parent-with-return :transient t)
    ("fm" "manual setup in suffix" ts-parent-with-setup-suffix :transient t)
    ("fi" "mixing interactive" ts-interactive-basic :transient t)
    ("fe" "early return" ts-simple-messager :transient t)]]

   [["Managing State" ; padded right group
    ("sb" "a bunch of infixes" ts-basic-infixes :transient t)
    ("sc" "using scope (accepts prefix)" ts-scope :transient t)
    ("sn" "set & save / snowcones" ts-snowcone-eater :transient t)
    ("sp" "history key / ping-pong" ts-ping :transient t)
    ("sg" "always forget / goldfish" ts-goldfish :transient t)
    ("se" "always remember / elephant" ts-elephant :transient t)
    ("sd" "default values" ts-default-values :transient t)
    ("sf" "enforcing inputs" ts-enforcing-inputs :transient t)
    ("sl" "lisp variables" ts-lisp-variable :transient t)]

  ["CLI arguments"
    ("cb" "basic arguments" ts-switches-and-arguments :transient t)
    ("cm" "random-init infix" ts-maybe-on :transient t)
    ("cc" "basic choices" ts-animal-choices :transient t)
    ("ce" "exclusive switches" ts-exclusive-switches :transient t)
    ("ci" "incompatible switches" ts-incompatible :transient t)
    ("co" "completions for choices" ts-choices-with-completions :transient t)
    ("cc"  "cowsay cli wrapper" ts-cowsay :transient t)]]

   [["Visibility"
     ;; padded description to sc
    ("vp" "predicates                  " ts-visibility-predicates :transient t)
    ("vi" "inapt (not suitable)" ts-inapt-parent :transient t)
    ("vl" "levels" ts-levels-and-visibility :transient t)]

   ["Advanced"
    ("ac" "generated child" ts-generated-child :transient t)
    ("ag" "generated group" ts-generated-group :transient t)
    ("ai" "custom infixes" ts-inception :transient t)
    ("au" "custom infixes & update" ts-inception-update :transient t)]])

(provide 'transient-showcase)
;;; transient-showcase.el ends here

About

Example forms for transient UI's in Emacs

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Emacs Lisp 100.0%