Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Transaction folding #255

Open
wants to merge 2 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
75 changes: 74 additions & 1 deletion ledger-occur.el
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@

(require 'cl-lib)
(require 'ledger-navigate)
(require 'hideshow)

(defconst ledger-occur-overlay-property-name 'ledger-occur-custom-buffer-grep)

Expand Down Expand Up @@ -119,7 +120,9 @@ long, otherwise it is the word at point."
"Make an overlay for an invisible portion of the buffer, from BEG to END."
(let ((ovl (make-overlay beg end)))
(overlay-put ovl ledger-occur-overlay-property-name t)
(overlay-put ovl 'invisible t)))
(overlay-put ovl 'invisible t)
;; required to not display ... when using together with transaction folding
(overlay-put ovl 'display "")))

(defun ledger-occur-create-overlays (ovl-bounds)
"Create the overlays for the visible transactions.
Expand Down Expand Up @@ -173,6 +176,76 @@ BUFFER-MATCHES should be a list of (BEG END) lists."
(setq current-end (cadr match))))
(nreverse (push (list current-beginning current-end) points)))))

;; -----------------------------------------------------------------------------
;; transactions folding
;; FIXME: probably this is not the best file to include this functionality.
;; -----------------------------------------------------------------------------
(defvar-local ledger-mode-folding-transactions-hidden nil
"Whether transactions are globally hidden or not.")

(defvar-local ledger-mode-toggle-invisible-transactions t
"Toggle invisible transactions (see ledger-occur-mode).")

(defun ledger-mode-transaction-toggle-folding ()
"Toggle hiding of transaction block under point.
A transaction block is identified as in ledger-highlight-xact-under-point.

Overlay of type `code' is used so that hidden blocks are
temporarily opened when doing incremental search."
(interactive)
(if (not (and (boundp 'hs-minor-mode) hs-minor-mode))
(message "Enable hs-minor-mode to use this functionality.")
(let ((exts (ledger-navigate-find-element-extents (point))))
(let ((b (car exts))
(e (cadr exts))
(p (point)))
(when (and (> (- e b) 1) ; not an empty line
(<= p e) (>= p b) ; point is within the boundaries
(not (region-active-p))) ; no active region
(save-excursion
(goto-char b)
(goto-char (line-end-position))
(if (hs-overlay-at (point)) ;; if transaction is hidden show it
(hs-show-block)
(goto-char b)
(hs-discard-overlays (line-end-position) e)
(hs-make-overlay (line-end-position) e 'code)
(run-hooks 'hs-hide-hook))))))))

(defun ledger-mode-request-toggle-transaction-hiding-p ()
"Decide whether to request transaction folding.
Assume that point is at the first transaction delimiter."
(goto-char (line-end-position))
(let ((ov-hs (hs-overlay-at (point)))
(ovs (overlays-at (point)))
(to_request t))
(when (not (eq (not ledger-mode-folding-transactions-hidden)
(overlayp ov-hs)))
;; handle behavior in invisible regions (see ledger-occur-mode)
(when (not ledger-mode-toggle-invisible-transactions)
(dolist (ov ovs)
(when (and (overlay-get ov 'invisible)
(overlay-get ov ledger-occur-overlay-property-name))
(setq to_request nil))))
to_request)))

(defun ledger-mode-folding-toggle-transactions ()
"Toggle hiding of all transactions in the current buffer."
(interactive)
(if (not (and (boundp 'hs-minor-mode) hs-minor-mode))
(message "Enable hs-minor-mode to use this functionality.")
(hs-life-goes-on
(save-excursion
(goto-char (point-min))
(while (re-search-forward "^[=~[:digit:]]" nil t)
(beginning-of-line)
(when (ledger-mode-request-toggle-transaction-hiding-p)
(ledger-mode-transaction-toggle-folding)
(goto-char (line-end-position))))
(setq ledger-mode-folding-transactions-hidden
(not ledger-mode-folding-transactions-hidden))))))
;; -----------------------------------------------------------------------------

(provide 'ledger-occur)

;;; ledger-occur.el ends here
44 changes: 44 additions & 0 deletions test/occur-test.el
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,50 @@ https://github.com/ledger/ledger-mode/issues/415"
Income:Salary
"))))

(ert-deftest ledger-occur/test-005 ()
"Test transaction folding"
:tags '(transaction folding)

(ledger-tests-with-temp-file
"2011/01/02 Grocery Store
Expenses:Food:Groceries $ 65.00
* Assets:Checking

2011/01/05 Employer
* Assets:Checking $ 2000.00
Income:Salary
"
(progn
(hs-minor-mode t)
(ledger-mode-folding-toggle-transactions))
(should
(equal (ledger-test-visible-buffer-string)
"2011/01/02 Grocery Store

2011/01/05 Employer
"))))

(ert-deftest ledger-occur/test-006 ()
"Test transaction folding together with ledger-occur-mode"
:tags '(transaction folding)

(ledger-tests-with-temp-file
"2011/01/02 Grocery Store
Expenses:Food:Groceries $ 65.00
* Assets:Checking

2011/01/05 Employer
* Assets:Checking $ 2000.00
Income:Salary
"
(progn
(hs-minor-mode t)
(ledger-occur "Groceries")
(ledger-mode-folding-toggle-transactions))
(should
(equal (ledger-test-visible-buffer-string)
"2011/01/02 Grocery Store
"))))

(provide 'occur-test)

Expand Down