123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781 |
- ;;; org-cycle.el --- Visibility cycling of Org entries -*- lexical-binding: t; -*-
- ;;
- ;; Copyright (C) 2020-2020 Free Software Foundation, Inc.
- ;;
- ;; Maintainer: Ihor Radchenko <yantar92 at gmail dot com>
- ;; Keywords: folding, visibility cycling, invisible text
- ;; URL: https://orgmode.org
- ;;
- ;; This file is part of GNU Emacs.
- ;;
- ;; GNU Emacs is free software: you can redistribute it and/or modify
- ;; it under the terms of the GNU General Public License as published by
- ;; the Free Software Foundation, either version 3 of the License, or
- ;; (at your option) any later version.
- ;; GNU Emacs is distributed in the hope that it will be useful,
- ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
- ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- ;; GNU General Public License for more details.
- ;; You should have received a copy of the GNU General Public License
- ;; along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
- ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
- ;;
- ;;; Commentary:
- ;; This file contains code controlling global folding state in buffer
- ;; and TAB-cycling.
- ;;; Code:
- (require 'org-macs)
- (org-assert-version)
- (require 'org-macs)
- (require 'org-fold)
- (declare-function org-element-type "org-element" (element))
- (declare-function org-element-property "org-element" (property element))
- (declare-function org-element-lineage "org-element" (datum &optional types with-self))
- (declare-function org-element-at-point "org-element" (&optional pom cached-only))
- (declare-function org-get-tags "org" (&optional pos local fontify))
- (declare-function org-subtree-end-visible-p "org" ())
- (declare-function org-narrow-to-subtree "org" (&optional element))
- (declare-function org-at-property-p "org" ())
- (declare-function org-re-property "org" (property &optional literal allow-null value))
- (declare-function org-item-beginning-re "org" ())
- (declare-function org-at-heading-p "org" (&optional invisible-not-ok))
- (declare-function org-at-item-p "org" ())
- (declare-function org-before-first-heading-p "org" ())
- (declare-function org-back-to-heading "org" (&optional invisible-ok))
- (declare-function org-end-of-subtree "org" (&optional invisible-ok to-heading))
- (declare-function org-entry-end-position "org" ())
- (declare-function org-try-cdlatex-tab "org" ())
- (declare-function org-cycle-level "org" ())
- (declare-function org-table-next-field "org-table" ())
- (declare-function org-table-justify-field-maybe "org-table" (&optional new))
- (declare-function org-inlinetask-at-task-p "org-inlinetask" ())
- (declare-function org-inlinetask-toggle-visibility "org-inlinetask" ())
- (declare-function org-list-get-all-items "org-list" (item struct prevs))
- (declare-function org-list-get-bottom-point "org-list" (struct))
- (declare-function org-list-prevs-alist "org-list" (struct))
- (declare-function org-list-set-item-visibility "org-list" (item struct view))
- (declare-function org-list-search-forward "org-list" (regexp &optional bound noerror))
- (declare-function org-list-has-child-p "org-list" (item struct))
- (declare-function org-list-get-item-end-before-blank "org-list" (item struct))
- (declare-function org-list-struct "org-list" ())
- (declare-function org-cycle-item-indentation "org-list" ())
- (declare-function outline-previous-heading "outline" ())
- (declare-function outline-next-heading "outline" ())
- (declare-function outline-end-of-heading "outline" ())
- (declare-function outline-up-heading "outline" (arg &optional invisible-ok))
- (defvar org-drawer-regexp)
- (defvar org-odd-levels-only)
- (defvar org-startup-folded)
- (defvar org-archive-tag)
- (defvar org-cycle-include-plain-lists)
- (defvar org-outline-regexp-bol)
- (defvar-local org-cycle-global-status nil)
- (put 'org-cycle-global-status 'org-state t)
- (defvar-local org-cycle-subtree-status nil)
- (put 'org-cycle-subtree-status 'org-state t)
- ;;;; Customisation:
- (defgroup org-cycle nil
- "Options concerning visibility cycling in Org mode."
- :tag "Org Cycle"
- :group 'org-structure)
- (defcustom org-cycle-skip-children-state-if-no-children t
- "Non-nil means skip CHILDREN state in entries that don't have any."
- :group 'org-cycle
- :type 'boolean)
- (defcustom org-cycle-max-level nil
- "Maximum level which should still be subject to visibility cycling.
- Levels higher than this will, for cycling, be treated as text, not a headline.
- When `org-odd-levels-only' is set, a value of N in this variable actually
- means 2N-1 stars as the limiting headline.
- When nil, cycle all levels.
- Note that the limiting level of cycling is also influenced by
- `org-inlinetask-min-level'. When `org-cycle-max-level' is not set but
- `org-inlinetask-min-level' is, cycling will be limited to levels one less
- than its value."
- :group 'org-cycle
- :type '(choice
- (const :tag "No limit" nil)
- (integer :tag "Maximum level")))
- (defcustom org-cycle-hide-block-startup nil
- "Non-nil means entering Org mode will fold all blocks.
- This can also be set in on a per-file basis with
- #+STARTUP: hideblocks
- #+STARTUP: nohideblocks"
- :group 'org-startup
- :group 'org-cycle
- :type 'boolean)
- (defcustom org-cycle-hide-drawer-startup t
- "Non-nil means entering Org mode will fold all drawers.
- This can also be set in on a per-file basis with
- #+STARTUP: hidedrawers
- #+STARTUP: nohidedrawers"
- :group 'org-startup
- :group 'org-cycle
- :type 'boolean)
- (defcustom org-cycle-global-at-bob nil
- "Cycle globally if cursor is at beginning of buffer and not at a headline.
- This makes it possible to do global cycling without having to use `S-TAB'
- or `\\[universal-argument] TAB'. For this special case to work, the first \
- line of the buffer
- must not be a headline -- it may be empty or some other text.
- When used in this way, `org-cycle-hook' is disabled temporarily to make
- sure the cursor stays at the beginning of the buffer.
- When this option is nil, don't do anything special at the beginning of
- the buffer."
- :group 'org-cycle
- :type 'boolean)
- (defcustom org-cycle-level-after-item/entry-creation t
- "Non-nil means cycle entry level or item indentation in new empty entries.
- When the cursor is at the end of an empty headline, i.e., with only stars
- and maybe a TODO keyword, TAB will then switch the entry to become a child,
- and then all possible ancestor states, before returning to the original state.
- This makes data entry extremely fast: M-RET to create a new headline,
- on TAB to make it a child, two or more tabs to make it a (grand-)uncle.
- When the cursor is at the end of an empty plain list item, one TAB will
- make it a subitem, two or more tabs will back up to make this an item
- higher up in the item hierarchy."
- :group 'org-cycle
- :type 'boolean)
- (defcustom org-cycle-emulate-tab t
- "Where should `org-cycle' emulate TAB.
- nil Never
- white Only in completely white lines
- whitestart Only at the beginning of lines, before the first non-white char
- t Everywhere except in headlines
- exc-hl-bol Everywhere except at the start of a headline
- If TAB is used in a place where it does not emulate TAB, the current subtree
- visibility is cycled."
- :group 'org-cycle
- :type '(choice (const :tag "Never" nil)
- (const :tag "Only in completely white lines" white)
- (const :tag "Before first char in a line" whitestart)
- (const :tag "Everywhere except in headlines" t)
- (const :tag "Everywhere except at bol in headlines" exc-hl-bol)))
- (defcustom org-cycle-separator-lines 2
- "Number of empty lines needed to keep an empty line between collapsed trees.
- If you leave an empty line between the end of a subtree and the following
- headline, this empty line is hidden when the subtree is folded.
- Org mode will leave (exactly) one empty line visible if the number of
- empty lines is equal or larger to the number given in this variable.
- So the default 2 means at least 2 empty lines after the end of a subtree
- are needed to produce free space between a collapsed subtree and the
- following headline.
- If the number is negative, and the number of empty lines is at least -N,
- all empty lines are shown.
- Special case: when 0, never leave empty lines in collapsed view."
- :group 'org-cycle
- :type 'integer)
- (put 'org-cycle-separator-lines 'safe-local-variable 'integerp)
- (defcustom org-cycle-pre-hook nil
- "Hook that is run before visibility cycling is happening.
- The function(s) in this hook must accept a single argument which indicates
- the new state that will be set right after running this hook. The
- argument is a symbol. Before a global state change, it can have the values
- `overview', `content', or `all'. Before a local state change, it can have
- the values `folded', `children', or `subtree'."
- :group 'org-cycle
- :type 'hook)
- (defcustom org-cycle-hook '(org-cycle-hide-archived-subtrees
- org-cycle-show-empty-lines
- org-cycle-optimize-window-after-visibility-change)
- "Hook that is run after `org-cycle' has changed the buffer visibility.
- The function(s) in this hook must accept a single argument which indicates
- the new state that was set by the most recent `org-cycle' command. The
- argument is a symbol. After a global state change, it can have the values
- `overview', `contents', or `all'. After a local state change, it can have
- the values `folded', `children', or `subtree'."
- :group 'org-cycle
- :package-version '(Org . "9.4")
- :type 'hook)
- (defcustom org-cycle-open-archived-trees nil
- "Non-nil means `org-cycle' will open archived trees.
- An archived tree is a tree marked with the tag ARCHIVE.
- When nil, archived trees will stay folded. You can still open them with
- normal outline commands like `show-all', but not with the cycling commands."
- :group 'org-archive
- :group 'org-cycle
- :type 'boolean)
- (defvar org-cycle-tab-first-hook nil
- "Hook for functions to attach themselves to TAB.
- See `org-ctrl-c-ctrl-c-hook' for more information.
- This hook runs as the first action when TAB is pressed, even before
- `org-cycle' messes around with the `outline-regexp' to cater for
- inline tasks and plain list item folding.
- If any function in this hook returns t, any other actions that
- would have been caused by TAB (such as table field motion or visibility
- cycling) will not occur.")
- ;;;; Implementation:
- (defun org-cycle-hide-drawers (state)
- "Re-hide all drawers after a visibility state change.
- STATE should be one of the symbols listed in the docstring of
- `org-cycle-hook'."
- (when (derived-mode-p 'org-mode)
- (cond ((not (memq state '(overview folded contents)))
- (let* ((global? (eq state 'all))
- (beg (if global? (point-min) (line-beginning-position)))
- (end (cond (global? (point-max))
- ((eq state 'children) (org-entry-end-position))
- (t (save-excursion (org-end-of-subtree t t))))))
- (org-fold--hide-drawers beg end)))
- ((memq state '(overview contents))
- ;; Hide drawers before first heading.
- (let ((beg (point-min))
- (end (save-excursion
- (goto-char (point-min))
- (if (org-before-first-heading-p)
- (org-entry-end-position)
- (point-min)))))
- (when (< beg end)
- (org-fold--hide-drawers beg end)))))))
- ;;;###autoload
- (defun org-cycle (&optional arg)
- "TAB-action and visibility cycling for Org mode.
- This is the command invoked in Org mode by the `TAB' key. Its main
- purpose is outline visibility cycling, but it also invokes other actions
- in special contexts.
- When this function is called with a `\\[universal-argument]' prefix, rotate \
- the entire
- buffer through 3 states (global cycling)
- 1. OVERVIEW: Show only top-level headlines.
- 2. CONTENTS: Show all headlines of all levels, but no body text.
- 3. SHOW ALL: Show everything.
- With a `\\[universal-argument] \\[universal-argument]' prefix argument, \
- switch to the startup visibility,
- determined by the variable `org-startup-folded', and by any VISIBILITY
- properties in the buffer.
- With a `\\[universal-argument] \\[universal-argument] \
- \\[universal-argument]' prefix argument, show the entire buffer, including
- any drawers.
- When inside a table, re-align the table and move to the next field.
- When point is at the beginning of a headline, rotate the subtree started
- by this line through 3 different states (local cycling)
- 1. FOLDED: Only the main headline is shown.
- 2. CHILDREN: The main headline and the direct children are shown.
- From this state, you can move to one of the children
- and zoom in further.
- 3. SUBTREE: Show the entire subtree, including body text.
- If there is no subtree, switch directly from CHILDREN to FOLDED.
- When point is at the beginning of an empty headline and the variable
- `org-cycle-level-after-item/entry-creation' is set, cycle the level
- of the headline by demoting and promoting it to likely levels. This
- speeds up creation document structure by pressing `TAB' once or several
- times right after creating a new headline.
- When there is a numeric prefix, go up to a heading with level ARG, do
- a `show-subtree' and return to the previous cursor position. If ARG
- is negative, go up that many levels.
- When point is not at the beginning of a headline, execute the global
- binding for `TAB', which is re-indenting the line. See the option
- `org-cycle-emulate-tab' for details.
- As a special case, if point is at the very beginning of the buffer, if
- there is no headline there, and if the variable `org-cycle-global-at-bob'
- is non-nil, this function acts as if called with prefix argument \
- \(`\\[universal-argument] TAB',
- same as `S-TAB') also when called without prefix argument."
- (interactive "P")
- (org-load-modules-maybe)
- (unless (or (run-hook-with-args-until-success 'org-cycle-tab-first-hook)
- (and org-cycle-level-after-item/entry-creation
- (or (org-cycle-level)
- (org-cycle-item-indentation))))
- (let* ((limit-level
- (or org-cycle-max-level
- (and (boundp 'org-inlinetask-min-level)
- org-inlinetask-min-level
- (1- org-inlinetask-min-level))))
- (nstars
- (and limit-level
- (if org-odd-levels-only
- (1- (* 2 limit-level))
- limit-level)))
- (org-outline-regexp
- (format "\\*%s " (if nstars (format "\\{1,%d\\}" nstars) "+"))))
- (cond
- ((equal arg '(16))
- (setq last-command 'dummy)
- (org-cycle-set-startup-visibility)
- (org-unlogged-message "Startup visibility, plus VISIBILITY properties"))
- ((equal arg '(64))
- (org-fold-show-all)
- (org-unlogged-message "Entire buffer visible, including drawers"))
- ((equal arg '(4)) (org-cycle-internal-global))
- ;; Show-subtree, ARG levels up from here.
- ((integerp arg)
- (save-excursion
- (org-back-to-heading)
- (outline-up-heading (if (< arg 0) (- arg)
- (- (funcall outline-level) arg)))
- (org-fold-show-subtree)))
- ;; Global cycling at BOB: delegate to `org-cycle-internal-global'.
- ((and org-cycle-global-at-bob
- (bobp)
- (not (looking-at org-outline-regexp)))
- (let ((org-cycle-hook
- (remq 'org-cycle-optimize-window-after-visibility-change
- org-cycle-hook)))
- (org-cycle-internal-global)))
- ;; Try CDLaTeX TAB completion.
- ((org-try-cdlatex-tab))
- ;; Inline task: delegate to `org-inlinetask-toggle-visibility'.
- ((and (featurep 'org-inlinetask)
- (org-inlinetask-at-task-p)
- (or (bolp) (not (eq org-cycle-emulate-tab 'exc-hl-bol))))
- (org-inlinetask-toggle-visibility))
- (t
- (let ((pos (point))
- (element (org-element-at-point)))
- (cond
- ;; Try toggling visibility for block at point.
- ((org-fold-hide-block-toggle nil t element))
- ;; Try toggling visibility for drawer at point.
- ((org-fold-hide-drawer-toggle nil t element))
- ;; Table: enter it or move to the next field.
- ((and (org-match-line "[ \t]*[|+]")
- (org-element-lineage element '(table) t))
- (if (and (eq 'table (org-element-type element))
- (eq 'table.el (org-element-property :type element)))
- (message (substitute-command-keys "\\<org-mode-map>\
- Use `\\[org-edit-special]' to edit table.el tables"))
- (org-table-justify-field-maybe)
- (call-interactively #'org-table-next-field)))
- ((run-hook-with-args-until-success
- 'org-tab-after-check-for-table-hook))
- ;; At an item/headline: delegate to `org-cycle-internal-local'.
- ((and (or (and org-cycle-include-plain-lists
- (let ((item (org-element-lineage element
- '(item plain-list)
- t)))
- (and item
- (= (line-beginning-position)
- (org-element-property :post-affiliated
- item)))))
- (org-match-line org-outline-regexp))
- (or (bolp) (not (eq org-cycle-emulate-tab 'exc-hl-bol))))
- (org-cycle-internal-local))
- ;; From there: TAB emulation and template completion.
- (buffer-read-only (org-back-to-heading))
- ((run-hook-with-args-until-success
- 'org-tab-after-check-for-cycling-hook))
- ((run-hook-with-args-until-success
- 'org-tab-before-tab-emulation-hook))
- ((and (eq org-cycle-emulate-tab 'exc-hl-bol)
- (or (not (bolp))
- (not (looking-at org-outline-regexp))))
- (call-interactively (global-key-binding (kbd "TAB"))))
- ((or (eq org-cycle-emulate-tab t)
- (and (memq org-cycle-emulate-tab '(white whitestart))
- (save-excursion (beginning-of-line 1) (looking-at "[ \t]*"))
- (or (and (eq org-cycle-emulate-tab 'white)
- (= (match-end 0) (line-end-position)))
- (and (eq org-cycle-emulate-tab 'whitestart)
- (>= (match-end 0) pos)))))
- (call-interactively (global-key-binding (kbd "TAB"))))
- (t
- (save-excursion
- (org-back-to-heading)
- (org-cycle))))))))))
- (defun org-cycle-force-archived ()
- "Cycle subtree even if it is archived."
- (interactive)
- (setq this-command 'org-cycle)
- (let ((org-cycle-open-archived-trees t))
- (call-interactively 'org-cycle)))
- (defun org-cycle-internal-global ()
- "Do the global cycling action."
- ;; Hack to avoid display of messages for .org attachments in Gnus
- (let ((ga (string-match-p "\\*fontification" (buffer-name))))
- (cond
- ((and (eq last-command this-command)
- (eq org-cycle-global-status 'overview))
- ;; We just created the overview - now do table of contents
- ;; This can be slow in very large buffers, so indicate action
- (run-hook-with-args 'org-cycle-pre-hook 'contents)
- (unless ga (org-unlogged-message "CONTENTS..."))
- (org-cycle-content)
- (unless ga (org-unlogged-message "CONTENTS...done"))
- (setq org-cycle-global-status 'contents)
- (run-hook-with-args 'org-cycle-hook 'contents))
- ((and (eq last-command this-command)
- (eq org-cycle-global-status 'contents))
- ;; We just showed the table of contents - now show everything
- (run-hook-with-args 'org-cycle-pre-hook 'all)
- (org-fold-show-all '(headings blocks))
- (unless ga (org-unlogged-message "SHOW ALL"))
- (setq org-cycle-global-status 'all)
- (run-hook-with-args 'org-cycle-hook 'all))
- (t
- ;; Default action: go to overview
- (run-hook-with-args 'org-cycle-pre-hook 'overview)
- (org-cycle-overview)
- (unless ga (org-unlogged-message "OVERVIEW"))
- (setq org-cycle-global-status 'overview)
- (run-hook-with-args 'org-cycle-hook 'overview)))))
- (defun org-cycle-internal-local ()
- "Do the local cycling action."
- (let ((goal-column 0) eoh eol eos has-children children-skipped struct)
- ;; First, determine end of headline (EOH), end of subtree or item
- ;; (EOS), and if item or heading has children (HAS-CHILDREN).
- (save-excursion
- (if (org-at-item-p)
- (progn
- (beginning-of-line)
- (setq struct (org-list-struct))
- (setq eoh (line-end-position))
- (setq eos (org-list-get-item-end-before-blank (point) struct))
- (setq has-children (org-list-has-child-p (point) struct)))
- (org-back-to-heading)
- (setq eoh (save-excursion (outline-end-of-heading) (point)))
- (setq eos (save-excursion
- (org-end-of-subtree t t)
- (unless (eobp) (forward-char -1))
- (point)))
- (setq has-children
- (or
- (save-excursion
- (let ((level (funcall outline-level)))
- (outline-next-heading)
- (and (org-at-heading-p)
- (> (funcall outline-level) level))))
- (and (eq org-cycle-include-plain-lists 'integrate)
- (save-excursion
- (org-list-search-forward (org-item-beginning-re) eos t))))))
- ;; Determine end invisible part of buffer (EOL)
- (beginning-of-line 2)
- (if (eq org-fold-core-style 'text-properties)
- (while (and (not (eobp)) ;this is like `next-line'
- (org-fold-folded-p (1- (point))))
- (goto-char (org-fold-next-visibility-change nil nil t))
- (and (eolp) (beginning-of-line 2)))
- (while (and (not (eobp)) ;this is like `next-line'
- (get-char-property (1- (point)) 'invisible))
- (goto-char (next-single-char-property-change (point) 'invisible))
- (and (eolp) (beginning-of-line 2))))
- (setq eol (point)))
- ;; Find out what to do next and set `this-command'
- (cond
- ((= eos eoh)
- ;; Nothing is hidden behind this heading
- (unless (org-before-first-heading-p)
- (run-hook-with-args 'org-cycle-pre-hook 'empty))
- (org-unlogged-message "EMPTY ENTRY")
- (setq org-cycle-subtree-status nil)
- (save-excursion
- (goto-char eos)
- (org-with-limited-levels
- (outline-next-heading))
- (when (org-invisible-p) (org-fold-heading nil))))
- ((and (or (>= eol eos)
- (save-excursion (goto-char eol) (skip-chars-forward "[:space:]" eos) (= (point) eos)))
- (or has-children
- (not (setq children-skipped
- org-cycle-skip-children-state-if-no-children))))
- ;; Entire subtree is hidden in one line: children view
- (unless (org-before-first-heading-p)
- (org-with-limited-levels
- (run-hook-with-args 'org-cycle-pre-hook 'children)))
- (if (org-at-item-p)
- (org-list-set-item-visibility (line-beginning-position) struct 'children)
- (org-fold-show-entry)
- (org-with-limited-levels (org-fold-show-children))
- (org-fold-show-set-visibility 'tree)
- ;; Fold every list in subtree to top-level items.
- (when (eq org-cycle-include-plain-lists 'integrate)
- (save-excursion
- (org-back-to-heading)
- (while (org-list-search-forward (org-item-beginning-re) eos t)
- (beginning-of-line 1)
- (let* ((struct (org-list-struct))
- (prevs (org-list-prevs-alist struct))
- (end (org-list-get-bottom-point struct)))
- (dolist (e (org-list-get-all-items (point) struct prevs))
- (org-list-set-item-visibility e struct 'folded))
- (goto-char (if (< end eos) end eos)))))))
- (org-unlogged-message "CHILDREN")
- (save-excursion
- (goto-char eos)
- (org-with-limited-levels
- (outline-next-heading))
- (when (and
- ;; Subtree does not end at the end of visible section of the
- ;; buffer.
- (< (point) (point-max))
- (org-invisible-p))
- ;; Reveal the following heading line.
- (org-fold-heading nil)))
- (setq org-cycle-subtree-status 'children)
- (unless (org-before-first-heading-p)
- (run-hook-with-args 'org-cycle-hook 'children)))
- ((or children-skipped
- (and (eq last-command this-command)
- (eq org-cycle-subtree-status 'children)))
- ;; We just showed the children, or no children are there,
- ;; now show everything.
- (unless (org-before-first-heading-p)
- (run-hook-with-args 'org-pre-cycle-hook 'subtree))
- (org-fold-region eoh eos nil 'outline)
- (org-unlogged-message
- (if children-skipped "SUBTREE (NO CHILDREN)" "SUBTREE"))
- (setq org-cycle-subtree-status 'subtree)
- (unless (org-before-first-heading-p)
- (run-hook-with-args 'org-cycle-hook 'subtree)))
- (t
- ;; Default action: hide the subtree.
- (run-hook-with-args 'org-cycle-pre-hook 'folded)
- (org-fold-region eoh eos t 'outline)
- (org-unlogged-message "FOLDED")
- (setq org-cycle-subtree-status 'folded)
- (unless (org-before-first-heading-p)
- (run-hook-with-args 'org-cycle-hook 'folded))))))
- ;;;###autoload
- (defun org-cycle-global (&optional arg)
- "Cycle the global visibility. For details see `org-cycle'.
- With `\\[universal-argument]' prefix ARG, switch to startup visibility.
- With a numeric prefix, show all headlines up to that level."
- (interactive "P")
- (cond
- ((integerp arg)
- (org-cycle-content arg)
- (setq org-cycle-global-status 'contents))
- ((equal arg '(4))
- (org-cycle-set-startup-visibility)
- (org-unlogged-message "Startup visibility, plus VISIBILITY properties."))
- (t
- (org-cycle '(4)))))
- (defun org-cycle-set-startup-visibility ()
- "Set the visibility required by startup options and properties."
- (cond
- ((eq org-startup-folded t)
- (org-cycle-overview))
- ((eq org-startup-folded 'content)
- (org-cycle-content))
- ((eq org-startup-folded 'show2levels)
- (org-cycle-content 2))
- ((eq org-startup-folded 'show3levels)
- (org-cycle-content 3))
- ((eq org-startup-folded 'show4levels)
- (org-cycle-content 4))
- ((eq org-startup-folded 'show5levels)
- (org-cycle-content 5))
- ((or (eq org-startup-folded 'showeverything)
- (eq org-startup-folded nil))
- (org-fold-show-all)))
- (unless (eq org-startup-folded 'showeverything)
- (when org-cycle-hide-block-startup (org-fold-hide-block-all))
- (org-cycle-set-visibility-according-to-property)
- (org-cycle-hide-archived-subtrees 'all)
- (when org-cycle-hide-drawer-startup (org-cycle-hide-drawers 'all))
- (org-cycle-show-empty-lines t)))
- (defun org-cycle-set-visibility-according-to-property ()
- "Switch subtree visibility according to VISIBILITY property."
- (interactive)
- (let ((regexp (org-re-property "VISIBILITY")))
- (org-with-point-at 1
- (while (re-search-forward regexp nil t)
- (let ((state (match-string 3)))
- (if (not (org-at-property-p)) (outline-next-heading)
- (save-excursion
- (org-back-to-heading t)
- (org-fold-subtree t)
- (org-fold-reveal)
- (pcase state
- ("folded"
- (org-fold-subtree t))
- ("children"
- (org-fold-show-hidden-entry)
- (org-fold-show-children))
- ("content"
- (save-excursion
- (save-restriction
- (org-narrow-to-subtree)
- (org-cycle-content))))
- ((or "all" "showall")
- (org-fold-show-subtree))
- (_ nil)))
- (org-end-of-subtree)))))))
- (defun org-cycle-overview ()
- "Switch to overview mode, showing only top-level headlines."
- (interactive)
- (save-excursion
- (goto-char (point-min))
- ;; Hide top-level drawer.
- (save-restriction
- (narrow-to-region (point-min) (or (re-search-forward org-outline-regexp-bol nil t) (point-max)))
- (org-fold-hide-drawer-all))
- (goto-char (point-min))
- (when (re-search-forward org-outline-regexp-bol nil t)
- (let* ((last (line-end-position))
- (level (- (match-end 0) (match-beginning 0) 1))
- (regexp (format "^\\*\\{1,%d\\} " level)))
- (while (re-search-forward regexp nil :move)
- (org-fold-region last (line-end-position 0) t 'outline)
- (setq last (line-end-position))
- (setq level (- (match-end 0) (match-beginning 0) 1))
- (setq regexp (format "^\\*\\{1,%d\\} " level)))
- (org-fold-region last (point) t 'outline)))))
- (defun org-cycle-content (&optional arg)
- "Show all headlines in the buffer, like a table of contents.
- With numerical argument N, show content up to level N."
- (interactive "p")
- (org-fold-show-all '(headings))
- (save-excursion
- (goto-char (point-min))
- ;; Hide top-level drawer.
- (save-restriction
- (narrow-to-region (point-min) (or (re-search-forward org-outline-regexp-bol nil t) (point-max)))
- (org-fold-hide-drawer-all))
- (goto-char (point-max))
- (let ((regexp (if (and (wholenump arg) (> arg 0))
- (format "^\\*\\{1,%d\\} " arg)
- "^\\*+ "))
- (last (point)))
- (while (re-search-backward regexp nil t)
- (org-fold-region (line-end-position) last t 'outline)
- (setq last (line-end-position 0))))))
- (defvar org-cycle-scroll-position-to-restore nil
- "Temporarily store scroll position to restore.")
- (defun org-cycle-optimize-window-after-visibility-change (state)
- "Adjust the window after a change in outline visibility.
- This function is the default value of the hook `org-cycle-hook'."
- (when (get-buffer-window (current-buffer))
- (let ((repeat (eq last-command this-command)))
- (unless repeat
- (setq org-cycle-scroll-position-to-restore nil))
- (cond
- ((eq state 'content) nil)
- ((eq state 'all) nil)
- ((and org-cycle-scroll-position-to-restore repeat
- (eq state 'folded))
- (set-window-start nil org-cycle-scroll-position-to-restore))
- ((eq state 'folded) nil)
- ((eq state 'children)
- (setq org-cycle-scroll-position-to-restore (window-start))
- (or (org-subtree-end-visible-p) (recenter 1)))
- ((eq state 'subtree)
- (unless repeat
- (setq org-cycle-scroll-position-to-restore (window-start)))
- (or (org-subtree-end-visible-p) (recenter 1)))))))
- (defun org-cycle-show-empty-lines (state)
- "Show empty lines above all visible headlines.
- The region to be covered depends on STATE when called through
- `org-cycle-hook'. Lisp program can use t for STATE to get the
- entire buffer covered. Note that an empty line is only shown if there
- are at least `org-cycle-separator-lines' empty lines before the headline."
- (when (/= org-cycle-separator-lines 0)
- (save-excursion
- (let* ((n (abs org-cycle-separator-lines))
- (re (cond
- ((= n 1) "\\(\n[ \t]*\n\\*+\\) ")
- ((= n 2) "^[ \t]*\\(\n[ \t]*\n\\*+\\) ")
- (t (let ((ns (number-to-string (- n 2))))
- (concat "^\\(?:[ \t]*\n\\)\\{" ns "," ns "\\}"
- "[ \t]*\\(\n[ \t]*\n\\*+\\) ")))))
- beg end)
- (cond
- ((memq state '(overview contents t))
- (setq beg (point-min) end (point-max)))
- ((memq state '(children folded))
- (setq beg (point)
- end (progn (org-end-of-subtree t t)
- (line-beginning-position 2)))))
- (when beg
- (goto-char beg)
- (while (re-search-forward re end t)
- (unless (org-invisible-p (match-end 1))
- (let ((e (match-end 1))
- (b (if (>= org-cycle-separator-lines 0)
- (match-beginning 1)
- (save-excursion
- (goto-char (match-beginning 0))
- (skip-chars-backward " \t\n")
- (line-end-position)))))
- (org-fold-region b e nil 'outline))))))))
- ;; Never hide empty lines at the end of the file.
- (save-excursion
- (goto-char (point-max))
- (outline-previous-heading)
- (outline-end-of-heading)
- (when (and (looking-at "[ \t\n]+")
- (= (match-end 0) (point-max)))
- (org-fold-region (point) (match-end 0) nil 'outline))))
- (defun org-cycle-hide-archived-subtrees (state)
- "Re-hide all archived subtrees after a visibility state change.
- STATE should be one of the symbols listed in the docstring of
- `org-cycle-hook'."
- (when (and (not org-cycle-open-archived-trees)
- (not (memq state '(overview folded))))
- (let ((globalp (memq state '(contents all))))
- (if globalp
- (org-fold-hide-archived-subtrees (point-min) (point-max))
- (org-fold-hide-archived-subtrees
- (point)
- (save-excursion
- (org-end-of-subtree t))))
- (when (and (not globalp)
- (member org-archive-tag
- (org-get-tags nil 'local)))
- (message "%s" (substitute-command-keys
- "Subtree is archived and stays closed. Use \
- `\\[org-cycle-force-archived]' to cycle it anyway."))))))
- (provide 'org-cycle)
- ;;; org-cycle.el ends here
|