summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJohn Wiegley <johnw@newartisans.com>2006-03-11 11:52:24 +0000
committerJohn Wiegley <johnw@newartisans.com>2008-04-13 02:41:30 -0400
commit49ae3b65d5bb8ca0482ccaeb88f4b0917016587f (patch)
tree0a9c2d73daf005e7fe523d8c11415105ce125ec1
parent47e2a341764b1747a0b010b3304233f7643f2a70 (diff)
downloadfork-ledger-49ae3b65d5bb8ca0482ccaeb88f4b0917016587f.tar.gz
fork-ledger-49ae3b65d5bb8ca0482ccaeb88f4b0917016587f.tar.bz2
fork-ledger-49ae3b65d5bb8ca0482ccaeb88f4b0917016587f.zip
*** empty log message ***
-rw-r--r--ledger.el228
-rw-r--r--main.cc2
-rw-r--r--valexpr.cc179
-rw-r--r--valexpr.h9
4 files changed, 352 insertions, 66 deletions
diff --git a/ledger.el b/ledger.el
index 71f9c183..450f0e8e 100644
--- a/ledger.el
+++ b/ledger.el
@@ -39,11 +39,35 @@
;; C-c C-y set default year for entry mode
;; C-c C-m set default month for entry mode
;; C-c C-r reconcile uncleared entries related to an account
+;; C-c C-o C-r run a ledger report
+;; C-C C-o C-g goto the ledger report buffer
+;; C-c C-o C-e edit the defined ledger reports
+;; C-c C-o C-s save a report definition based on the current report
+;; C-c C-o C-a rerun a ledger report
+;; C-c C-o C-k kill the ledger report buffer
;;
;; In the reconcile buffer, use SPACE to toggle the cleared status of
;; a transaction, C-x C-s to save changes (to the ledger file as
;; well), or C-c C-r to attempt an auto-reconcilation based on the
;; statement's ending date and balance.
+;;
+;; The ledger reports command asks the user to select a report to run
+;; then creates a report buffer containing the results of running the
+;; associated command line. Its' behavior is modified by a prefix
+;; argument which, when given, causes the generated command line that
+;; will be used to create the report to be presented for editing
+;; before the report is actually run. Arbitrary unnamed command lines
+;; can be run by specifying an empty name for the report. The command
+;; line used can later be named and saved for future use as a named
+;; report from the generated reports buffer.
+;;
+;; In a report buffer, the following keys are available:
+;; (space) scroll up
+;; e edit the defined ledger reports
+;; s save a report definition based on the current report
+;; q quit the report (return to ledger buffer)
+;; r redo the report
+;; k kill the report buffer
(require 'esh-util)
(require 'esh-arg)
@@ -65,6 +89,16 @@
:type 'boolean
:group 'ledger)
+(defcustom ledger-reports
+ '(("bal" "ledger bal")
+ ("reg" "ledger reg"))
+ "Definition of reports to run.
+
+Each element has the form (NAME CMDLINE)"
+ :type '(repeat (list (string :tag "Report Name")
+ (string :tag "Command Line")))
+ :group 'ledger)
+
(defvar bold 'bold)
(defvar ledger-font-lock-keywords
'(("^[0-9./=]+\\s-+\\(?:([^)]+)\\s-+\\)?\\([^*].+\\)" 1 bold)
@@ -324,7 +358,18 @@ dropped."
(define-key map [(control ?c) (control ?y)] 'ledger-set-year)
(define-key map [(control ?c) (control ?m)] 'ledger-set-month)
(define-key map [(control ?c) (control ?c)] 'ledger-toggle-current)
- (define-key map [(control ?c) (control ?r)] 'ledger-reconcile)))
+ (define-key map [(control ?c) (control ?r)] 'ledger-reconcile)
+ (define-key map [(control ?c) (control ?o) (control ?r)] 'ledger-report)
+ (define-key map [(control ?c) (control ?o) (control ?g)]
+ 'ledger-report-goto)
+ (define-key map [(control ?c) (control ?o) (control ?a)]
+ 'ledger-report-redo)
+ (define-key map [(control ?c) (control ?o) (control ?s)]
+ 'ledger-report-save)
+ (define-key map [(control ?c) (control ?o) (control ?e)]
+ 'ledger-report-edit)
+ (define-key map [(control ?c) (control ?o) (control ?k)]
+ 'ledger-report-kill)))
;; Reconcile mode
@@ -540,6 +585,187 @@ dropped."
(define-key map [?q] 'ledger-reconcile-quit)
(use-local-map map)))
+;; Ledger report mode
+
+(defvar ledger-report-buffer-name "*Ledger Report*")
+
+(defvar ledger-report-name nil)
+(defvar ledger-report-cmd nil)
+(defvar ledger-report-name-prompt-history nil)
+(defvar ledger-report-cmd-prompt-history nil)
+(defvar ledger-original-window-cfg nil)
+
+(defvar ledger-report-mode-abbrev-table)
+
+(define-derived-mode ledger-report-mode text-mode "Ledger-Report"
+ "A mode for viewing ledger reports."
+ (let ((map (make-sparse-keymap)))
+ (define-key map [? ] 'scroll-up)
+ (define-key map [?r] 'ledger-report-redo)
+ (define-key map [?s] 'ledger-report-save)
+ (define-key map [?k] 'ledger-report-kill)
+ (define-key map [?e] 'ledger-report-edit)
+ (define-key map [?q] 'ledger-report-quit)
+ (define-key map [(control ?c) (control ?l) (control ?r)]
+ 'ledger-report-redo)
+ (define-key map [(control ?c) (control ?l) (control ?S)]
+ 'ledger-report-save)
+ (define-key map [(control ?c) (control ?l) (control ?k)]
+ 'ledger-report-kill)
+ (define-key map [(control ?c) (control ?l) (control ?e)]
+ 'ledger-report-edit)
+ (use-local-map map)))
+
+(defun ledger-report-read-name ()
+ "Read the name of a ledger report to use, with completion.
+
+The empty string and unknown names are allowed."
+ (completing-read "Report name: "
+ ledger-reports nil nil nil
+ 'ledger-report-name-prompt-history nil))
+
+(defun ledger-report (report-name edit)
+ "Run a user-specified report from `ledger-reports'.
+
+Prompts the user for the name of the report to run. If no name is
+entered, the user will be prompted for a command line to run. The
+command line specified or associated with the selected report name
+is run and the output is made available in another buffer for viewing.
+If a prefix argument is given and the user selects a valid report
+name, the user is prompted with the corresponding command line for
+editing before the command is run.
+
+The output buffer will be in `ledger-report-mode', which defines
+commands for saving a new named report based on the command line
+used to generate the buffer, navigating the buffer, etc."
+ (interactive
+ (let ((rname (ledger-report-read-name))
+ (edit (not (null current-prefix-arg))))
+ (list rname edit)))
+ (let ((buf (current-buffer))
+ (rbuf (get-buffer ledger-report-buffer-name))
+ (wcfg (current-window-configuration)))
+ (if rbuf
+ (kill-buffer rbuf))
+ (with-current-buffer
+ (pop-to-buffer (get-buffer-create ledger-report-buffer-name))
+ (ledger-report-mode)
+ (set (make-local-variable 'ledger-buf) buf)
+ (set (make-local-variable 'ledger-report-name) report-name)
+ (set (make-local-variable 'ledger-original-window-cfg) wcfg)
+ (ledger-do-report (ledger-report-cmd report-name edit))
+ (shrink-window-if-larger-than-buffer))))
+
+(defun string-empty-p (s)
+ "Check for the empty string."
+ (string-equal "" s))
+
+(defun ledger-report-name-exists (name)
+ "Check to see if the given report name exists.
+
+If name exists, returns the object naming the report, otherwise returns nil."
+ (unless (string-empty-p name)
+ (car (assoc name ledger-reports))))
+
+(defun ledger-reports-add (name cmd)
+ "Add a new report to `ledger-reports'."
+ (setq ledger-reports (cons (list name cmd) ledger-reports)))
+
+(defun ledger-reports-custom-save ()
+ "Save the `ledger-reports' variable using the customize framework."
+ (customize-save-variable 'ledger-reports ledger-reports))
+
+(defun ledger-report-read-command (report-cmd)
+ "Read the command line to create a report."
+ (read-from-minibuffer "Report command line: "
+ (if (null report-cmd) "ledger " report-cmd)
+ nil nil 'ledger-report-cmd-prompt-history))
+
+(defun ledger-report-cmd (report-name edit)
+ "Get the command line to run the report."
+ (let ((report-cmd (car (cdr (assoc report-name ledger-reports)))))
+ ;; logic for substitution goes here
+ (when (or (null report-cmd) edit)
+ (setq report-cmd (ledger-report-read-command report-cmd)))
+ (set (make-local-variable 'ledger-report-cmd) report-cmd)
+ (or (string-empty-p report-name)
+ (ledger-report-name-exists report-name)
+ (ledger-reports-add report-name report-cmd)
+ (ledger-reports-custom-save))
+ report-cmd))
+
+(defun ledger-do-report (cmd)
+ "Run a report command line."
+ (goto-char (point-min))
+ (insert (format "Report: %s\n" cmd)
+ (make-string (- (window-width) 1) ?=)
+ "\n")
+ (shell-command cmd t nil))
+
+(defun ledger-report-goto ()
+ "Goto the ledger report buffer."
+ (interactive)
+ (let ((rbuf (get-buffer ledger-report-buffer-name)))
+ (if (not rbuf)
+ (error "There is no ledger report buffer"))
+ (pop-to-buffer rbuf)
+ (shrink-window-if-larger-than-buffer)))
+
+(defun ledger-report-redo ()
+ "Redo the report in the current ledger report buffer."
+ (interactive)
+ (ledger-report-goto)
+ (erase-buffer)
+ (ledger-do-report ledger-report-cmd))
+
+(defun ledger-report-quit ()
+ "Quit the ledger report buffer."
+ (interactive)
+ (ledger-report-goto)
+ (set-window-configuration ledger-original-window-cfg))
+
+(defun ledger-report-kill ()
+ "Kill the ledger report buffer."
+ (interactive)
+ (ledger-report-quit)
+ (kill-buffer (get-buffer ledger-report-buffer-name)))
+
+(defun ledger-report-edit ()
+ "Edit the defined ledger reports."
+ (interactive)
+ (customize-variable 'ledger-reports))
+
+(defun ledger-report-read-new-name ()
+ "Read the name for a new report from the minibuffer."
+ (let ((name ""))
+ (while (string-empty-p name)
+ (setq name (read-from-minibuffer "Report name: " nil nil nil
+ 'ledger-report-name-prompt-history)))
+ name))
+
+(defun ledger-report-save ()
+ "Save the current report command line as a named report."
+ (interactive)
+ (ledger-report-goto)
+ (let (existing-name)
+ (when (string-empty-p ledger-report-name)
+ (setq ledger-report-name (ledger-report-read-new-name)))
+
+ (while (setq existing-name (ledger-report-name-exists ledger-report-name))
+ (cond ((y-or-n-p (format "Overwrite existing report named '%s' "
+ ledger-report-name))
+ (when (string-equal
+ ledger-report-cmd
+ (car (cdr (assq existing-name ledger-reports))))
+ (error "Current command is identical to existing saved one"))
+ (setq ledger-reports
+ (assq-delete-all existing-name ledger-reports)))
+ (t
+ (setq ledger-report-name (ledger-report-read-new-name)))))
+
+ (ledger-reports-add ledger-report-name ledger-report-cmd)
+ (ledger-reports-custom-save)))
+
;; A sample function for $ users
(defun ledger-align-dollars (&optional column)
diff --git a/main.cc b/main.cc
index caed1ab6..387524ba 100644
--- a/main.cc
+++ b/main.cc
@@ -249,7 +249,7 @@ int parse_and_report(config_t& config, int argc, char * argv[], char * envp[])
ledger::dump_value_expr(std::cout, expr.get());
std::cout << std::endl;
std::cout << "Value expression parsed was:" << std::endl;
- ledger::write_value_expr(std::cout, expr.get(), NULL, 0);
+ ledger::write_value_expr(std::cout, expr.get());
std::cout << std::endl << std::endl;
std::cout << "Result of computation: ";
}
diff --git a/valexpr.cc b/valexpr.cc
index 00ab8f87..76213506 100644
--- a/valexpr.cc
+++ b/valexpr.cc
@@ -1516,21 +1516,35 @@ void valexpr_context::describe(std::ostream& out) const throw()
out << desc << std::endl;
out << " ";
- unsigned long start = out.tellp();
- unsigned long pos = ledger::write_value_expr(out, expr,
- error_node, start);
- out << std::endl << " ";
- for (int i = 0; i < pos - start; i++)
- out << " ";
- out << "^" << std::endl;
+ unsigned long start = (long)out.tellp() - 1;
+ unsigned long begin;
+ unsigned long end;
+ bool found = ledger::write_value_expr(out, expr, error_node, &begin, &end);
+ out << std::endl;
+ if (found) {
+ out << " ";
+ for (int i = 0; i < end - start; i++) {
+ if (i >= begin - start)
+ out << "^";
+ else
+ out << " ";
+ }
+ out << std::endl;
+ }
}
-unsigned long write_value_expr(std::ostream& out,
- const value_expr_t * node,
- const value_expr_t * node_to_find,
- unsigned long start_pos)
+bool write_value_expr(std::ostream& out,
+ const value_expr_t * node,
+ const value_expr_t * node_to_find,
+ unsigned long * start_pos,
+ unsigned long * end_pos)
{
- long pos = start_pos;
+ bool found = false;
+
+ if (start_pos && node == node_to_find) {
+ *start_pos = (long)out.tellp() - 1;
+ found = true;
+ }
switch (node->kind) {
case value_expr_t::CONSTANT_I:
@@ -1566,62 +1580,74 @@ unsigned long write_value_expr(std::ostream& out,
case value_expr_t::F_ARITH_MEAN:
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_ABS: out << "abs"; break;
out << "abs(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_QUANTITY: out << "quantity"; break;
out << "quantity(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_COMMODITY: out << "commodity"; break;
out << "commodity(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_SET_COMMODITY: out << "set_commodity"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_VALUE: out << "valueof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_PRICE: out << "priceof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_DATE: out << "dateof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_DATECMP: out << "datecmp"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_YEAR: out << "yearof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_MONTH: out << "monthof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::F_DAY: out << "dayof"; break;
out << "average(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
@@ -1646,15 +1672,18 @@ unsigned long write_value_expr(std::ostream& out,
case value_expr_t::O_NOT:
out << "!";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
break;
case value_expr_t::O_NEG:
out << "-";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
break;
case value_expr_t::O_PERC:
out << "%";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
break;
case value_expr_t::O_ARG:
@@ -1667,107 +1696,137 @@ unsigned long write_value_expr(std::ostream& out,
break;
case value_expr_t::O_COM:
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << ", ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
break;
case value_expr_t::O_QUES:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " ? ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_COL:
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " : ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
break;
case value_expr_t::O_AND: out << "O_AND"; break;
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " & ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_OR: out << "O_OR"; break;
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " | ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_NEQ:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " != ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_EQ:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " == ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_LT:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " < ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_LTE:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " <= ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_GT:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " > ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_GTE:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " >= ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_ADD:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " + ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_SUB:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " - ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_MUL:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " * ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
case value_expr_t::O_DIV:
out << "(";
- pos = write_value_expr(out, node->left, node_to_find, pos);
+ if (write_value_expr(out, node->left, node_to_find, start_pos, end_pos))
+ found = true;
out << " / ";
- pos = write_value_expr(out, node->right, node_to_find, pos);
+ if (write_value_expr(out, node->right, node_to_find, start_pos, end_pos))
+ found = true;
out << ")";
break;
@@ -1777,10 +1836,10 @@ unsigned long write_value_expr(std::ostream& out,
break;
}
- if (node == node_to_find)
- pos = (long)out.tellp() - 1;
+ if (end_pos && node == node_to_find)
+ *end_pos = (long)out.tellp() - 1;
- return pos;
+ return found;
}
void dump_value_expr(std::ostream& out, const value_expr_t * node,
diff --git a/valexpr.h b/valexpr.h
index 461a8497..3d260d43 100644
--- a/valexpr.h
+++ b/valexpr.h
@@ -322,10 +322,11 @@ inline value_expr_t * parse_value_expr(const char * p,
void dump_value_expr(std::ostream& out, const value_expr_t * node,
const int depth = 0);
-unsigned long write_value_expr(std::ostream& out,
- const value_expr_t * node,
- const value_expr_t * node_to_find = NULL,
- unsigned long start_pos = 0UL);
+bool write_value_expr(std::ostream& out,
+ const value_expr_t * node,
+ const value_expr_t * node_to_find = NULL,
+ unsigned long * start_pos = NULL,
+ unsigned long * end_pos = NULL);
//////////////////////////////////////////////////////////////////////