- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 196
Hybrid S expressions
While direct manipulation of S-expressions (sexps, balanced expressions) is especially rewarding in languages like elisp, scheme, clojure and friends, in most "regular" C-like languages (C, Java, C#, ...), the notion of "sexp" is very weak for useful manipulation of source code.
To partially solve this smartparens introduces a notion of "hybrid sexp". First discussed in Issue #95, a general framework for retrieving and manipulating hybrid sexps is being added into smartparens.
A definition of hybrid sexp is as follows: it is the smallest balanced region containing the point that does not expand further than current line, with the only exception to keep the balance. Here, by balance we mean any hanging opened sexps that are not closed on the same line. Here are some examples. A () pair will represent sexp delimiters while [] will represent the hybrid sexp. As usual, | represents point.
Note: the definition is really clumsy but I really have no better idea on explaining it. If you understand what's going on and can provide better description, please start an issue and tell me! (or email me)
[foo ba|r baz]
quux flux
;;---
[foo ba|r (baz
           quux)] flux
;;---
([foo b|ar baz]) quux flux
;;---
(foo
 [bar b|az]) quux flux
;;---
([foo|]
 bar baz) quux fluxHere's a list of functions that can operate on hybrid sexps, with brief descriptions. You should read the documentation in emacs via C-h f function-name RET to get more information.
sp-kill-hybrid-sexp (arg)
sp-transpose-hybrid-sexp (&optional arg)
sp-push-hybrid-sexp ()
sp-slurp-hybrid-sexp ()
sp-indent-adjust-sexp ()
sp-dedent-adjust-sexp ()- 
sp-kill-hybrid-sexp- Kill a line as if withkill-line, but respecting delimiters. This works almost the same asparedit-kill.
- 
sp-transpose-hybrid-sexp- Transpose the hybrid line the point is at with the one before it.sp-backward-sexpis used to "travel" into the previous hybrid sexp.
- 
sp-push-hybrid-sexp- Similar to translate, but instead of transposing previous expression, it pushes the following one over the next.
- 
sp-slurp-hybrid-sexp- Likesp-forward-slurp-sexpbut slurps a hybrid sexps. At the moment does not support any prefix arguments.
- 
sp-indent-adjust-sexp- Add the hybrid sexp at point to the previous S-expression
- 
sp-dedent-adjust-sexp- Barf everything after the point. This is almost the same assp-forward-slurp-sexpbut handles hanging delimiters in a special way.
We will now illustrate these functions on examples. In all examples | marks the point
for | (int i = 1; i < 10) {
  bar = i + 1;
}
  :=>
for |
//---
|int baz = (bar
            /
            2);
  :=>
|for (int i = 1; i < 10) {
  bar = i + 1;
}
int |baz = (bar
            /
            2);
  :=>
int baz = (bar
           /
           2);
for (int i = 1; i < 10) {
  bar = i + 1;
}
||for (int i = 1; i < 10) {
  bar = i + 1;
}
int baz = (bar
           /
           2);
  :=>
|int baz = (bar
            /
            2);
for (int i = 1; i < 10) {
  bar = i + 1;
}for (int i = 1; i < 10) {
 |bar = i + 1;
}
int baz = (bar
           /
           2);
return 0;
  :=>
for (int i = 1; i < 10) {
 |bar = i + 1;
  int baz = (bar
             /
             2);
}
return 0;
  :=>
for (int i = 1; i < 10) {
 |bar = i + 1;
  int baz = (bar
             /
             2);
  return 0;
}