elisp: SMIE Indentation Example

 
 22.7.1.8 Sample Indentation Rules
 .................................
 
 Here is an example of an indentation function:
 
      (defun sample-smie-rules (kind token)
        (pcase (cons kind token)
          (`(:elem . basic) sample-indent-basic)
          (`(,_ . ",") (smie-rule-separator kind))
          (`(:after . ":=") sample-indent-basic)
          (`(:before . ,(or `"begin" `"(" `"{")))
           (if (smie-rule-hanging-p) (smie-rule-parent)))
          (`(:before . "if")
           (and (not (smie-rule-bolp)) (smie-rule-prev-p "else")
                (smie-rule-parent)))))
 
 A few things to note:
 
    • The first case indicates the basic indentation increment to use.
      If ‘sample-indent-basic’ is ‘nil’, then SMIE uses the global
      setting ‘smie-indent-basic’.  The major mode could have set
      ‘smie-indent-basic’ buffer-locally instead, but that is
      discouraged.
 
    • The rule for the token ‘","’ make SMIE try to be more clever when
      the comma separator is placed at the beginning of lines.  It tries
      to outdent the separator so as to align the code after the comma;
      for example:
 
           x = longfunctionname (
                   arg1
                 , arg2
               );
 
    • The rule for indentation after ‘":="’ exists because otherwise SMIE
      would treat ‘":="’ as an infix operator and would align the right
      argument with the left one.
 
    • The rule for indentation before ‘"begin"’ is an example of the use
      of virtual indentation: This rule is used only when ‘"begin"’ is
      hanging, which can happen only when ‘"begin"’ is not at the
      beginning of a line.  So this is not used when indenting ‘"begin"’
      itself but only when indenting something relative to this
      ‘"begin"’.  Concretely, this rule changes the indentation from:
 
               if x > 0 then begin
                       dosomething(x);
                   end
      to
               if x > 0 then begin
                   dosomething(x);
               end
 
    • The rule for indentation before ‘"if"’ is similar to the one for
      ‘"begin"’, but where the purpose is to treat ‘"else if"’ as a
      single unit, so as to align a sequence of tests rather than indent
      each test further to the right.  This function does this only in
      the case where the ‘"if"’ is not placed on a separate line, hence
      the ‘smie-rule-bolp’ test.
 
      If we know that the ‘"else"’ is always aligned with its ‘"if"’ and
      is always at the beginning of a line, we can use a more efficient
      rule:
           ((equal token "if")
            (and (not (smie-rule-bolp))
                 (smie-rule-prev-p "else")
                 (save-excursion
                   (sample-smie-backward-token)
                   (cons 'column (current-column)))))
 
      The advantage of this formulation is that it reuses the indentation
      of the previous ‘"else"’, rather than going all the way back to the
      first ‘"if"’ of the sequence.