;;; increment.el -- Increment numbers at point -*- lexical-binding: t; -*- (require 'cl-macs) (defvar increment--binary-number-regexp (rx (group (or ?- word-start)) "0b" (group (* ?0)) (group (+ (any "01"))) word-end)) (defvar increment--octal-number-regexp (rx (group (or ?- word-start)) "0o" (group (* ?0)) (group (+ (any "0-7"))) word-end)) (defvar increment--decimal-number-regexp (rx (group (? ?-)) (group (* ?0)) (group (+ (any digit))))) (defvar increment--hexadecimal-number-regexp (rx (group (or ?- word-start)) "0x" (group (* ?0)) (group (+ (any hex-digit))) word-end)) (defvar increment--hexadecimal-lower-number-regexp (rx (group (or ?- word-start)) "0o" (group (* ?0)) (group (+ (any "0-9a-f"))) word-end)) (defvar increment--number-regexp (rx (or (seq (or ?- word-start) (or (seq "0b" (+ (any "01"))) (seq "0o" (+ (any "0-7"))) (seq "0x" (+ (any hex-digit)))) word-end) (seq (? ?-) (+ (any digit)))))) (defun increment--number-to-binary-string (number) (let (s) (while (not (= number 0)) (setq s (concat (if (= 1 (logand number 1)) "1" "0") s) number (lsh number -1))) (when (string= s "") (setq s "0")) s)) (defun increment--format-number-with-base (number base leading-zeros buffer-substr hex-style) (let* ((neg (> 0 number)) (number (abs number)) (number-string (cond ((= base 2) (increment--number-to-binary-string number)) ((= base 8) (format "%o" number)) ((= base 10) (number-to-string number)) ((= base 16) (format (if (eq hex-style 'lower) "%x" "%X") number)) ((t (error (format "Invalid base %d") base))))) (length-diff (- (length buffer-substr) (length number-string))) (leading-zeros (if (> leading-zeros 0) (+ leading-zeros length-diff) 0))) (concat (when neg "-") (cond ((= base 2) "0b") ((= base 8) "0o") ((= base 16) "0x")) (when (> leading-zeros 0) (make-string leading-zeros ?0)) number-string))) (defun increment--match-number-at-point () (cond ((thing-at-point-looking-at increment--binary-number-regexp) (cons 2 nil)) ((thing-at-point-looking-at increment--octal-number-regexp) (cons 8 nil)) ((thing-at-point-looking-at increment--hexadecimal-number-regexp) (cons 16 nil)) ((thing-at-point-looking-at increment--hexadecimal-lower-number-regexp) (cons 16 'lower)) ((thing-at-point-looking-at increment--decimal-number-regexp) (cons 10 nil)))) ;;;###autoload (cl-defun increment-number-at-point (&optional arg) "Increment the number at point by ARG or 1 if ARG is nil. If called interactively, the universal argument can be used to specify ARG. If the number at point has leading zeros then the width of the number is preserved." (interactive "*p") (save-match-data (let (case-fold-search (match-pair (increment--match-number-at-point))) (unless match-pair (let ((save-point (point))) (unless (re-search-forward increment--number-regexp (line-end-position) :noerror) (goto-char save-point) (cl-return-from increment-number-at-point)) (setq match-pair (increment--match-number-at-point)))) (let* ((base (car match-pair)) (hex-style (cdr match-pair)) (substr (buffer-substring-no-properties (match-beginning 3) (match-end 3))) (sign (if (= (match-beginning 1) (match-end 1)) +1 -1)) (result (+ (* (string-to-number substr base) sign) (or arg 1)))) (replace-match (increment--format-number-with-base result base (- (match-end 2) (match-beginning 2)) substr hex-style)))))) ;;;###autoload (defun decrement-number-at-point (&optional arg) "The same as `increment-number-at-point', but ARG is negated." (interactive "*p") (increment-number-at-point (- (or arg 1)))) (provide 'increment)