(define-library (picrin pretty-print)
  (import (scheme base)
          (scheme write))

  ; (reverse-string-append l) = (apply string-append (reverse l))

  (define (reverse-string-append l)

    (define (rev-string-append l i)
      (if (pair? l)
          (let* ((str (car l))
                 (len (string-length str))
                 (result (rev-string-append (cdr l) (+ i len))))
            (let loop ((j 0) (k (- (- (string-length result) i) len)))
              (if (< j len)
                  (begin
                    (string-set! result k (string-ref str j))
                    (loop (+ j 1) (+ k 1)))
                  result)))
          (make-string i)))

    (rev-string-append l 0))

  ;; We define a pretty printer for Scheme S-expressions (sexp). While
  ;; Petite Scheme supports that by its own, mzscheme does not. If you
  ;; get a sexp (like from proof-to-expr) prefix it with a call to spp and
  ;; the output is nicely formated to fit into pp-width many columns:
  ;;
  ;;  (spp (proof-to-expr (current-proof)))
  ;;

  (define pp-width 80)

  ;;"genwrite.scm" generic write used by pretty-print and truncated-print.
  ;; Copyright (c) 1991, Marc Feeley
  ;; Author: Marc Feeley (feeley@iro.umontreal.ca)
  ;; Distribution restrictions: none
  ;;
  ;; Modified for Minlog by Stefan Schimanski <schimans@math.lmu.de>
  ;; Taken from slib 2d6, genwrite.scm and pp.scm

  (define genwrite:newline-str (make-string 1 #\newline))

  (define (generic-write obj display? width output)

    (define (read-macro? l)
      (define (length1? l) (and (pair? l) (null? (cdr l))))
      (let ((head (car l)) (tail (cdr l)))
        (case head
          ((quote quasiquote unquote unquote-splicing) (length1? tail))
          (else                                        #f))))

    (define (read-macro-body l)
      (cadr l))

    (define (read-macro-prefix l)
      (let ((head (car l)) (tail (cdr l)))
        (case head
          ((quote)            "'")
          ((quasiquote)       "`")
          ((unquote)          ",")
          ((unquote-splicing) ",@"))))

    (define (out str col)
      (and col (output str) (+ col (string-length str))))

    (define (wr obj col)

      (define (wr-lst l col)
        (if (pair? l)
            (let loop ((l (cdr l))
                       (col (and col (wr (car l) (out "(" col)))))
              (cond ((not col) col)
                    ((pair? l)
                     (loop (cdr l) (wr (car l) (out " " col))))
                    ((null? l) (out ")" col))
                    (else      (out ")" (wr l (out " . " col))))))
            (out "()" col)))

      (define (wr-expr expr col)
        (if (read-macro? expr)
            (wr (read-macro-body expr) (out (read-macro-prefix expr) col))
            (wr-lst expr col)))

      (cond ((pair? obj)        (wr-expr obj col))
            ((null? obj)        (wr-lst obj col))
            ((vector? obj)      (wr-lst (vector->list obj) (out "#" col)))
            ((boolean? obj)     (out (if obj "#t" "#f") col))
            ((number? obj)      (out (number->string obj) col))
            ((symbol? obj)      (out (symbol->string obj) col))
            ((procedure? obj)   (out "#[procedure]" col))
            ((string? obj)      (if display?
                                    (out obj col)
                                    (let loop ((i 0) (j 0) (col (out "\"" col)))
                                      (if (and col (< j (string-length obj)))
                                          (let ((c (string-ref obj j)))
                                            (if (or (char=? c #\\)
                                                    (char=? c #\"))
                                                (loop j
                                                      (+ j 1)
                                                      (out "\\"
                                                           (out (substring obj i j)
                                                                col)))
                                                (loop i (+ j 1) col)))
                                          (out "\""
                                               (out (substring obj i j) col))))))
            ((char? obj)        (if display?
                                    (out (make-string 1 obj) col)
                                    (out (case obj
                                           ((#\space)   "space")
                                           ((#\newline) "newline")
                                           (else        (make-string 1 obj)))
                                         (out "#\\" col))))
            ((input-port? obj)  (out "#[input-port]" col))
            ((output-port? obj) (out "#[output-port]" col))
            ((eof-object? obj)  (out "#[eof-object]" col))
            (else               (out "#[unknown]" col))))

    (define (pp obj col)

      (define (spaces n col)
        (if (> n 0)
            (if (> n 7)
                (spaces (- n 8) (out "        " col))
                (out (substring "        " 0 n) col))
            col))

      (define (indent to col)
        (and col
             (if (< to col)
                 (and (out genwrite:newline-str col) (spaces to 0))
                 (spaces (- to col) col))))

      (define pp-list #f)
      (define pp-expr #f)
      (define pp-call #f)
      (define pp-down #f)
      (define pp-general #f)
      (define pp-width #f)
      (define pp-expr-list #f)

      (define indent-general #f)
      (define max-expr-width #f)
      (define max-call-head-width #f)
      (define style #f)

      (define pr
        (lambda (obj col extra pp-pair)
          (if (or (pair? obj) (vector? obj)) ; may have to split on multiple lines
              (let ((result '())
                    (left (min (+ (- (- width col) extra) 1) max-expr-width)))
                (generic-write obj display? #f
                               (lambda (str)
                                 (set! result (cons str result))
                                 (set! left (- left (string-length str)))
                                 (> left 0)))
                (if (> left 0)        ; all can be printed on one line
                    (out (reverse-string-append result) col)
                    (if (pair? obj)
                        (pp-pair obj col extra)
                        (pp-list (vector->list obj) (out "#" col) extra pp-expr))))
              (wr obj col))))

      (set! pp-expr
        (lambda (expr col extra)
          (if (read-macro? expr)
              (pr (read-macro-body expr)
                  (out (read-macro-prefix expr) col)
                  extra
                  pp-expr)
              (let ((head (car expr)))
                (if (symbol? head)
                    (let ((proc (style head)))
                      (if proc
                          (proc expr col extra)
                          (if (> (string-length (symbol->string head))
                                 max-call-head-width)
                              (pp-general expr col extra #f #f #f pp-expr)
                              (pp-call expr col extra pp-expr))))
                    (pp-list expr col extra pp-expr))))))

                                        ; (head item1
                                        ;       item2
                                        ;       item3)
      (set! pp-call
        (lambda (expr col extra pp-item)
          (let ((col* (wr (car expr) (out "(" col))))
            (and col
                 (pp-down (cdr expr) col* (+ col* 1) extra pp-item)))))

                                        ; (item1
                                        ;  item2
                                        ;  item3)
      (set! pp-list
        (lambda (l col extra pp-item)
          (let ((col (out "(" col)))
            (pp-down l col col extra pp-item))))

      (set! pp-down
        (lambda (l col1 col2 extra pp-item)
          (let loop ((l l) (col col1))
            (and col
                 (cond ((pair? l)
                        (let ((rest (cdr l)))
                          (let ((extra (if (null? rest) (+ extra 1) 0)))
                            (loop rest
                                  (pr (car l) (indent col2 col) extra pp-item)))))
                       ((null? l)
                        (out ")" col))
                       (else
                        (out ")"
                             (pr l
                                 (indent col2 (out "." (indent col2 col)))
                                 (+ extra 1)
                                 pp-item))))))))

      (set! pp-general
        (lambda (expr col extra named? pp-1 pp-2 pp-3)

          (define (tail3 rest col1 col2)
            (pp-down rest col2 col1 extra pp-3))

          (define (tail2 rest col1 col2 col3)
            (if (and pp-2 (pair? rest))
                (let* ((val1 (car rest))
                       (rest (cdr rest))
                       (extra (if (null? rest) (+ extra 1) 0)))
                  (tail3 rest col1 (pr val1 (indent col3 col2) extra pp-2)))
                (tail3 rest col1 col2)))

          (define (tail1 rest col1 col2 col3)
            (if (and pp-1 (pair? rest))
                (let* ((val1 (car rest))
                       (rest (cdr rest))
                       (extra (if (null? rest) (+ extra 1) 0)))
                  (tail2 rest col1 (pr val1 (indent col3 col2) extra pp-1) col3))
                (tail2 rest col1 col2 col3)))

          (let* ((head (car expr))
                 (rest (cdr expr))
                 (col* (wr head (out "(" col))))
            (if (and named? (pair? rest))
                (let* ((name (car rest))
                       (rest (cdr rest))
                       (col** (wr name (out " " col*))))
                  (tail1 rest (+ col indent-general) col** (+ col** 1)))
                (tail1 rest (+ col indent-general) col* (+ col* 1))))))

      (set! pp-expr-list
        (lambda (l col extra)
          (pp-list l col extra pp-expr)))

      (define (pp-LAMBDA expr col extra)
        (pp-general expr col extra #f pp-expr-list #f pp-expr))

      (define (pp-IF expr col extra)
        (pp-general expr col extra #f pp-expr #f pp-expr))

      (define (pp-COND expr col extra)
        (pp-call expr col extra pp-expr-list))

      (define (pp-CASE expr col extra)
        (pp-general expr col extra #f pp-expr #f pp-expr-list))

      (define (pp-AND expr col extra)
        (pp-call expr col extra pp-expr))

      (define (pp-LET expr col extra)
        (let* ((rest (cdr expr))
               (named? (and (pair? rest) (symbol? (car rest)))))
          (pp-general expr col extra named? pp-expr-list #f pp-expr)))

      (define (pp-BEGIN expr col extra)
        (pp-general expr col extra #f #f #f pp-expr))

      (define (pp-DO expr col extra)
        (pp-general expr col extra #f pp-expr-list pp-expr-list pp-expr))

                                        ; define formatting style (change these to suit your style)

      (set! indent-general 2)

      (set! max-call-head-width 5)

      (set! max-expr-width 50)

      (set! style
        (lambda (head)
          (case head
            ((lambda let* letrec define) pp-LAMBDA)
            ((if set!)                   pp-IF)
            ((cond)                      pp-COND)
            ((case)                      pp-CASE)
            ((and or)                    pp-AND)
            ((let)                       pp-LET)
            ((begin)                     pp-BEGIN)
            ((do)                        pp-DO)
            (else                        #f))))

      (pr obj col 0 pp-expr))

    (if width
        (out genwrite:newline-str (pp obj 0))
        (wr obj 0)))

  (define (pretty-print obj . opt)
    (let ((port (if (pair? opt) (car opt) (current-output-port))))
      (generic-write obj #f pp-width
                     (lambda (s) (display s port) #t))
      (display "")))

  (export pretty-print))