Skip to content
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
04dcd3a
rewrite keymap system
mahmoodsh36 Jan 14, 2026
eb57d77
introduce keymap-activate and initialize transient.lisp
mahmoodsh36 Jan 14, 2026
7a213e7
add simple popup
mahmoodsh36 Jan 14, 2026
e3f6d6a
make popup more informative
mahmoodsh36 Jan 18, 2026
a3d56b3
improve docstrings a little more
mahmoodsh36 Jan 18, 2026
d8ac6fb
slight refactor
mahmoodsh36 Jan 18, 2026
fdb7bf9
make multi-choice prefix work
mahmoodsh36 Jan 20, 2026
2d7e130
fix order of keymaps (hence fix priorities)
mahmoodsh36 Jan 23, 2026
8fe748d
slight change
mahmoodsh36 Jan 23, 2026
7b9f3a6
add find-prefix-by-id
mahmoodsh36 Jan 23, 2026
42977a4
introduce :back, :cancel suffix values
mahmoodsh36 Jan 23, 2026
7e73c23
rename package to lem/transient
mahmoodsh36 Jan 25, 2026
ebce66e
rename package to lem/transient
mahmoodsh36 Jan 25, 2026
ba5694b
rename package to lem/transient and system to lem-transient
mahmoodsh36 Jan 25, 2026
33cdde7
use "intern" with explicit package name
mahmoodsh36 Jan 27, 2026
d5023be
introduce prefix-behavior, with-last-read-key-sequence, and handle ex…
mahmoodsh36 Jan 28, 2026
2397e14
use :description instead of :name for keymaps
mahmoodsh36 Jan 28, 2026
4d37d35
introduce toggle infix, use prompt for multi-choice infix
mahmoodsh36 Jan 29, 2026
8a418d8
fix vim search
mahmoodsh36 Jan 30, 2026
4b33545
fix undefine-key (actually undefine-key-internal)
mahmoodsh36 Jan 30, 2026
30d4ea5
remvoe some redundant stuff
mahmoodsh36 Jan 30, 2026
79229a8
properly parse keymap keywords in transient, change title styling
mahmoodsh36 Jan 30, 2026
3d8b2d5
small fix
mahmoodsh36 Feb 1, 2026
891b2f3
introduce "intermediate" prefixes
mahmoodsh36 Feb 3, 2026
3ea00ea
introduce *transient-always-show*
mahmoodsh36 Feb 4, 2026
1f3a109
small refactor
mahmoodsh36 Feb 4, 2026
facc26c
introduce multi-value infix to demo
mahmoodsh36 Feb 4, 2026
eb7fc25
remove outdated comment
mahmoodsh36 Feb 5, 2026
caf8743
introduce variable-syncing (sync infix with var)
mahmoodsh36 Feb 5, 2026
06b58e5
add :extend keyword to make-keymap
mahmoodsh36 Feb 7, 2026
f708a1e
add mode-transient-keymap, show extended keymap keys in transient
mahmoodsh36 Feb 7, 2026
0af3651
rename keymap-extend to keymap-base
mahmoodsh36 Feb 7, 2026
64631a3
export more symbols
mahmoodsh36 Feb 12, 2026
feafe41
remove redundant keymap-find-keybind arg and rename it to keymap-find
mahmoodsh36 Feb 13, 2026
d11647b
remove "dynamic properties" and just rely on CLOS
mahmoodsh36 Feb 13, 2026
50be8c8
small fixes
mahmoodsh36 Feb 14, 2026
42158ab
make parse-transient eval values, so that quotes are needed
mahmoodsh36 Feb 17, 2026
9ca5ad6
add bottomside-window and define transient-mode
mahmoodsh36 Feb 17, 2026
e3da420
add scrolling support
mahmoodsh36 Feb 18, 2026
ac9b675
fix horizontal scrolling
mahmoodsh36 Feb 18, 2026
c39df30
set always-show to nil
mahmoodsh36 Feb 18, 2026
a1533e3
remove "dirty" redrawing technique
mahmoodsh36 Feb 18, 2026
24badca
separate child keymaps from prefixes
mahmoodsh36 Feb 19, 2026
6e28acc
handle editor-abort in suffix prompts
mahmoodsh36 Feb 23, 2026
7ea0eda
add define-transient-key
mahmoodsh36 Feb 23, 2026
d811a8b
indentation
mahmoodsh36 Feb 24, 2026
6db325d
add define-prefix, add post-command hook to update popup
mahmoodsh36 Mar 9, 2026
46ebbbe
fix issue: self-insert was being invoked when it shouldnt have been
mahmoodsh36 Mar 9, 2026
1a42a5e
show maps that have show-p set, regardless of mode
mahmoodsh36 Mar 12, 2026
b69ae34
use defvar instead of defparameter to avoid issues on recomp
mahmoodsh36 Mar 17, 2026
5bda002
fix handling of undef-hook
mahmoodsh36 Mar 20, 2026
40ab039
set popup max-lines to 10 instead of 15 by default
mahmoodsh36 Mar 21, 2026
1610e48
fix undef-hook/function-table edge case
mahmoodsh36 Mar 26, 2026
62201d2
fix popup behavior when switching buffers
mahmoodsh36 Mar 29, 2026
9abe4ff
remove outdated comment
mahmoodsh36 Mar 29, 2026
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
remove "dynamic properties" and just rely on CLOS
the idea of "dynamic" properties/slots that i had before i think was
redundant and just overcomplicated things. this is a refactors things.
now we just rely on CLOS even for instance-specific "overrides" in
keymaps/prefixes.
  • Loading branch information
mahmoodsh36 committed Feb 13, 2026
commit d11647bcb5e91ef4e0e240872a01f6bd7a37d6ec
2 changes: 1 addition & 1 deletion extensions/copilot/copilot.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -246,7 +246,7 @@

Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code Contractor: internal_symbol_rule

Contract: contract

AI check failed: "internal_symbol_rule"

Reason:
Added code contains direct references to internal (non-exported) symbols via package double-colon qualifiers (e.g. lem-core::other-keymaps), which violates the rule to use exported symbols from lem or lem-core and avoid internal symbol access.

(defun find-copilot-completion-command (key)
(lookup-keybind key
:keymaps (append (lem-core::all-keymaps)
:keymaps (append (lem-core::other-keymaps)
(list *copilot-completion-keymap*))))

(defun search-preffix (str1 str2)
Expand Down
5 changes: 3 additions & 2 deletions extensions/transient/demo.lisp
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
(in-package :lem/transient)
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code Contractor: defpackage_rule

Contract: contract

AI check failed: "defpackage_rule"

Reason:
New files use IN-PACKAGE or other first forms instead of a top-level DEFPACKAGE or UIOP:DEFINE-PACKAGE as required.


(defvar *demo-language* "lisp"
(defvar *demo-language*
"lisp"
"a demo variable that stays in sync with an infix.")

(define-transient *demo-keymap*
Expand Down Expand Up @@ -39,7 +40,7 @@
(:key "l"
:type :choice
:id :mode
:choices ("lisp" "python" "js")
:choices-func (list "lisp" "python" "js")
:value "python"
:description "mode"))
(:keymap
Expand Down
134 changes: 67 additions & 67 deletions extensions/transient/keymap.lisp
Original file line number Diff line number Diff line change
Expand Up @@ -21,27 +21,7 @@
(:method ((mode mode))
nil))

(defmacro add-dynamic-property (class-name properties-accessor property-name &optional default-value)
"define <CLASS-NAME>-<PROPERTY-NAME> getter and setter methods.

the getter retrieves from PROPERTIES-ACCESSOR using :PROPERTY-NAME as key.
if the value is a function, it funcalls it. the setter stores directly.
if DEFAULT-VALUE is provided and non-nil, it is used as the default for getf."
(let* ((keyword (intern (symbol-name property-name) :keyword))
(getter-name (intern (format nil "~A-~A" class-name property-name) :lem/transient))
(obj-sym (gensym "OBJ")))
`(progn
(defmethod ,getter-name ((,obj-sym ,class-name))
(let ((prop ,(if default-value
`(getf (,properties-accessor ,obj-sym) ,keyword ,default-value)
`(getf (,properties-accessor ,obj-sym) ,keyword))))
(if (functionp prop)
(funcall prop)
prop)))
(defmethod (setf ,getter-name) (val (,obj-sym ,class-name))
(setf (getf (,properties-accessor ,obj-sym) ,keyword) val)))))

(defmacro add-static-property (class-name properties-accessor property-name &optional default-value)
(defmacro add-property (class-name properties-accessor property-name &optional default-value)
"define <CLASS-NAME>-<PROPERTY-NAME> getter and setter methods.

the getter retrieves from PROPERTIES-ACCESSOR using :PROPERTY-NAME as key.
Expand All @@ -57,15 +37,13 @@ the setter stores directly."
(defmethod (setf ,getter-name) (val (,obj-sym ,class-name))
(setf (getf (,properties-accessor ,obj-sym) ,keyword) val)))))

;; these are properties that we want to be "dynamic", as in can be assigned a function that
;; returns the value later instead of the value itself.
(add-dynamic-property keymap keymap-properties show-p nil)
(add-dynamic-property prefix prefix-properties show-p t)
;; static properties dont take a function that returns a value, just a value.
(add-static-property keymap keymap-properties display-style :row)
(add-static-property prefix prefix-properties id)
;; some stuff we need for working with "transient keymaps"
(add-property keymap keymap-properties show-p nil)
(add-property keymap keymap-properties display-style :row)
(add-property prefix prefix-properties show-p t)
(add-property prefix prefix-properties id)
;; TODO: it would be better to store the parsed key sequence instead of the stringified one and work with that.
(add-static-property prefix prefix-properties display-key)
(add-property prefix prefix-properties display-key)

(defun find-prefix-by-id (keymap id)
(labels ((f (node)
Expand Down Expand Up @@ -159,6 +137,27 @@ the setter stores directly."
(defmacro define-transient (name &body bindings)
`(defparameter ,name (parse-transient ',bindings)))

(defun parse-transient-method (object key val method-name)
(let* ((key-string (string key))
(key-method (intern (format nil "~A-~A" method-name key-string) :lem/transient))
(length (length key-string)))
(cond ((and (> length 5)
(string-equal "-func" (subseq key-string (- length 5))))
(let* ((prefix-key-string (subseq key-string 0 (- length 5)))
(key-method (intern (format nil "~A-~A" method-name prefix-key-string)
:lem/transient)))
(eval `(defmethod ,key-method ((object (eql ,object)))
,val))))
((fboundp key-method)
(funcall (fdefinition (list 'setf key-method)) val object))
(t
(let ((property-method (intern (format nil "~A-PROPERTIES" method-name)
:lem/transient)))
(when (fboundp property-method)
(let ((props (funcall (fdefinition property-method) object)))
(setf (getf props key) val)
(funcall (fdefinition (list 'setf property-method)) props object))))))))

(defun parse-transient (bindings)
"defines a transient menu. args yet to be documented."
(let ((keymap (make-keymap)))
Expand All @@ -170,12 +169,9 @@ the setter stores directly."
;; inline property
((keywordp binding)
(let ((val (second tail)))
(let ((key-method (intern (format nil "KEYMAP-~A" (string binding)) :lem/transient)))
(if (fboundp key-method)
(funcall (fdefinition (list 'setf key-method)) val keymap)
(setf (getf (keymap-properties keymap) binding) val))))
;; advance another cell because we're already consumed it (second tail)
(setf tail (cdr tail)))
(parse-transient-method keymap binding val "KEYMAP")
;; advance another cell because we're already consumed it (second tail)
(setf tail (cdr tail))))
;; direct child keymap (:keymap ...)
((eq (car binding) :keymap)
(let ((sub-map (parse-transient (cdr binding))))
Expand All @@ -191,40 +187,43 @@ the setter stores directly."
(when (cdr parsed-key)
(setf (prefix-display-key prefix) key))
;; we need to create intermediate prefixes if the key is longer than one
(loop for cell on parsed-key
for i from 0
for lastp = (null (cdr cell))
for current-key = (car cell)
for current-prefix = (if lastp
prefix
;; reuse existing intermediate prefix with same key, or create new one
(let ((existing (find current-key (keymap-children last-keymap)
:test (lambda (k child)
(and (typep child 'prefix)
(prefix-intermediate-p child)
(equal k (prefix-key child)))))))
(if existing
(progn
(setf last-keymap (prefix-suffix existing))
existing)
(let* ((new-prefix (make-instance 'prefix))
(new-keymap (make-keymap)))
(keymap-add-prefix last-keymap new-prefix t)
(setf (prefix-suffix new-prefix) new-keymap)
(setf (prefix-intermediate-p new-prefix) t)
(setf (keymap-show-p new-keymap) t)
(setf last-keymap new-keymap)
new-prefix))))
do (setf (prefix-key current-prefix) current-key))
(loop
for cell on parsed-key
for i from 0
for lastp = (null (cdr cell))
for current-key = (car cell)
do (let ((current-prefix
(if lastp
prefix
;; reuse existing intermediate prefix with same key, or create new one
(let ((existing (find
current-key
(keymap-children last-keymap)
:test (lambda (k child)
(and (typep child 'prefix)
(prefix-intermediate-p child)
(equal
k
(prefix-key child)))))))
(if existing
(progn
(setf last-keymap (prefix-suffix existing))
existing)
(let* ((new-prefix (make-instance 'prefix))
(new-keymap (make-keymap)))
(keymap-add-prefix last-keymap new-prefix t)
(setf (prefix-suffix new-prefix) new-keymap)
(setf (prefix-intermediate-p new-prefix) t)
(setf (keymap-show-p new-keymap) t)
(setf last-keymap new-keymap)
new-prefix))))))
(setf (prefix-key current-prefix) current-key)))
(keymap-add-prefix last-keymap prefix t)
;; sometimes the suffix will not be set (e.g. prefix-type is :choice). we
;; initialize it to nil to avoid unbound errors.
(setf (prefix-suffix prefix) nil)
(loop for (key value) on (cddr binding) by 'cddr
;; key-method is used for (setf prefix-<key-method> <value>)
for key-method = (intern (format nil "PREFIX-~A" (string key)) :lem/transient)
do (let ((setf-expr `(setf (,key-method prefix) value))
(final-value)
do (let ((final-value)
(should-set t))
(cond
;; if the suffix is a keymap we need to parse recursively
Expand All @@ -244,7 +243,8 @@ the setter stores directly."
(t
(setf final-value value)))
(when should-set
(funcall (fdefinition (list 'setf key-method))
final-value
prefix))))))))))
(parse-transient-method prefix
key
final-value
"PREFIX"))))))))))
keymap))
Loading
Loading