org-indent.el 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. ;; Need to hook into demotion, promotion functions: Redo the entire subtree
  2. ;; Problem: I have this huge list of overlays, should I be able
  3. ;; to reuse them?
  4. ; should be a mode...
  5. (defconst org-indent-strings (make-vector 80 nil))
  6. (loop for i from 0 to 79 do
  7. (aset org-indent-strings i (org-add-props (make-string i ?\ )
  8. nil 'face '(:underline nil :background "grey95"))))
  9. (defvar org-indent-overlays nil)
  10. (make-variable-buffer-local 'org-indent-overlays)
  11. (defun org-indent-initialize-buffer ()
  12. (interactive)
  13. (when (org-mode-p)
  14. (org-remove-indent-overlays)
  15. (let (n)
  16. (save-excursion
  17. (save-restriction
  18. (widen)
  19. (goto-char (point-min))
  20. (while (< (point) (point-max))
  21. (cond
  22. ((looking-at outline-regexp)
  23. (setq n (1+ (funcall outline-level))))
  24. (n
  25. (setq ov (org-make-overlay (1- (point)) (point)))
  26. (org-overlay-put ov 'after-string (aref org-indent-strings n))
  27. (org-overlay-put ov 'evaporate t)
  28. (org-overlay-put ov 'rear-sticky nil)
  29. (org-overlay-put ov 'front-sticky nil)
  30. (org-overlay-put ov 'org-indent n)
  31. (push ov org-indent-overlays)))
  32. (beginning-of-line 2)))))))
  33. (run-with-idle-timer 10 t 'org-indent-initialize-buffer)
  34. (defun org-indent-refresh (start end)
  35. "Refresh indentation overlays in the range given."
  36. (save-excursion
  37. (goto-char start)
  38. (when (search-forward "\n" end t)
  39. (goto-char start)
  40. (beginning-of-line 0)
  41. (let ((n (or (get-char-property (max (point-min) (1- (point)))
  42. 'org-indent) 0))
  43. ov e)
  44. (while (<= (point) end)
  45. (mapc (lambda (x) (if (org-overlay-get x 'org-indent)
  46. (if (> (setq e (org-overlay-end x)) (point))
  47. (org-move-overlay x (1- e) e)
  48. (org-delete-overlay x)
  49. (setq org-indent-overlays
  50. (delq x org-indent-overlays)))))
  51. (overlays-at (max (point-min) (1- (point)))))
  52. (cond
  53. ((looking-at outline-regexp)
  54. (setq n (1+ (funcall outline-level))))
  55. (n
  56. (setq ov (org-make-overlay (1- (point)) (point)))
  57. (org-overlay-put ov 'after-string (aref org-indent-strings n))
  58. (org-overlay-put ov 'evaporate t)
  59. (org-overlay-put ov 'rear-sticky nil)
  60. (org-overlay-put ov 'front-sticky nil)
  61. (org-overlay-put ov 'org-indent n)
  62. (push ov org-indent-overlays)))
  63. (beginning-of-line 2))))))
  64. (defun org-remove-indent-overlays ()
  65. (interactive)
  66. (mapc 'org-delete-overlay org-indent-overlays)
  67. (setq org-indent-overlays nil))
  68. (defun org-indent-after-change-function (start end length)
  69. "After change function for org-indent.
  70. Notices when an insert has added some lines and adjusts
  71. the line number extents accordingly."
  72. (if (= start end)
  73. () ;; this is a deletion
  74. (org-indent-refresh start end)))
  75. (add-hook 'after-change-functions 'org-indent-after-change-function)
  76. (defun org-fixup-indentation (diff)
  77. ""
  78. (save-excursion
  79. (let ((end (save-excursion (outline-next-heading)
  80. (point-marker)))
  81. (prohibit (if (> diff 0)
  82. "^\\S-"
  83. (concat "^ \\{0," (int-to-string (- diff)) "\\}\\S-")))
  84. col)
  85. (if (eq org-adapt-indentation 'virtual)
  86. (org-indent-refresh (point) end)
  87. (unless (save-excursion (end-of-line 1)
  88. (re-search-forward prohibit end t))
  89. (while (and (< (point) end)
  90. (re-search-forward "^[ \t]+" end t))
  91. (goto-char (match-end 0))
  92. (setq col (current-column))
  93. (if (< diff 0) (replace-match ""))
  94. (indent-to (+ diff col))))
  95. (move-marker end nil)))))
  96. (setq org-adapt-indentation 'virtual)