src/Pure/Syntax/pretty.ML
author clasohm
Thu, 16 Sep 1993 12:20:38 +0200
changeset 0 a5a9c433f639
child 18 c9ec452ff08f
permissions -rw-r--r--
Initial revision
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     1
(*  Title:      Pure/Syntax/pretty
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     2
    ID:         $Id$
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     3
    Author:     Lawrence C Paulson
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     4
    Copyright   1991  University of Cambridge
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     5
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     6
Pretty printing module
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     7
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     8
Loosely based on
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
     9
  D. C. Oppen, "Pretty Printing",
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    10
  ACM Transactions on Programming Languages and Systems (1980), 465-483.
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    11
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    12
The object to be printed is given as a tree with indentation and line
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    13
breaking information.  A "break" inserts a newline if the text until
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    14
the next break is too long to fit on the current line.  After the newline,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    15
text is indented to the level of the enclosing block.  Normally, if a block
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    16
is broken then all enclosing blocks will also be broken.  Only "inconsistent
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    17
breaks" are provided.
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    18
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    19
The stored length of a block is used in breakdist (to treat each inner block as
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    20
a unit for breaking).
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    21
*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    22
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    23
(* FIXME improve? *)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    24
type pprint_args = (string -> unit) * (int -> unit) * (int -> unit) * (unit -> unit);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    25
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    26
signature PRETTY =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    27
  sig
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    28
  type T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    29
  val blk : int * T list -> T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    30
  val brk : int -> T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    31
  val fbrk : T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    32
  val str : string -> T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    33
  val lst : string * string -> T list -> T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    34
  val quote : T -> T
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    35
  val string_of : T -> string
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    36
  val str_of : T -> string
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    37
  val pprint : T -> pprint_args -> unit
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    38
  val setdepth: int -> unit
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    39
  val setmargin: int -> unit
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    40
  end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    41
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    42
functor PrettyFun () : PRETTY =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    43
struct
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    44
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    45
(*Printing items: compound phrases, strings, and breaks*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    46
datatype T = Block of T list * int * int (*indentation, length*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    47
           | String of string
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    48
           | Break of bool*int   (*mandatory flag; width if not taken*);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    49
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    50
(*Add the lengths of the expressions until the next Break; if no Break then
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    51
  include "after", to account for text following this block. *)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    52
fun breakdist (Block(_,_,len)::es, after) = len + breakdist(es, after)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    53
  | breakdist (String s :: es, after) = size s + breakdist (es, after)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    54
  | breakdist (Break _ :: es, after) = 0
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    55
  | breakdist ([], after) = after;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    56
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    57
fun repstring a 0 = ""
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    58
  | repstring a 1 = a
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    59
  | repstring a k =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    60
      if k mod 2 = 0 then repstring(a^a) (k div 2)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    61
                     else repstring(a^a) (k div 2) ^ a;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    62
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    63
(*** Type for lines of text: string, # of lines, position on line ***)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    64
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    65
type text = {tx: string, nl: int, pos: int};
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    66
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    67
val emptytext = {tx="", nl=0, pos=0};
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    68
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    69
fun blanks wd {tx,nl,pos} =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    70
    {tx  = tx ^ repstring" "wd,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    71
     nl  = nl,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    72
     pos = pos+wd};
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    73
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    74
fun newline {tx,nl,pos} =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    75
    {tx  = tx ^ "\n",
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    76
     nl  = nl+1,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    77
     pos = 0};
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    78
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    79
fun string s {tx,nl,pos} =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    80
    {tx  = tx ^ s,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    81
     nl  = nl,
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    82
     pos = pos + size(s)};
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    83
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    84
(*** Formatting ***)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    85
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    86
val margin      = ref 80        (*right margin, or page width*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    87
and breakgain   = ref 4         (*minimum added space required of a break*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    88
and emergencypos = ref 40;      (*position too far to right*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    89
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    90
fun setmargin m =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    91
    (margin     := m;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    92
     breakgain  := !margin div 20;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    93
     emergencypos := !margin div 2);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    94
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    95
fun forcenext [] = []
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    96
  | forcenext (Break(_,wd) :: es) = Break(true,0) :: es
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    97
  | forcenext (e :: es) = e :: forcenext es;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    98
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
    99
fun format ([], _, _) text = text
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   100
  | format (e::es, blockin, after) (text as {pos,nl,...}) =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   101
    (case e of
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   102
       Block(bes,indent,wd) =>
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   103
         let val blockin' = (pos + indent) mod !emergencypos
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   104
             val btext = format(bes, blockin', breakdist(es,after)) text
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   105
             val es2 = if nl < #nl(btext) then forcenext es
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   106
                       else es
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   107
         in  format (es2,blockin,after) btext  end
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   108
     | String s => format (es,blockin,after) (string s text)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   109
     | Break(force,wd) => (* no break if text fits on this line
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   110
                      or if breaking would add only breakgain to space *)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   111
           format (es,blockin,after)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   112
               (if not force andalso
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   113
                   pos+wd <= max[!margin - breakdist(es,after),
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   114
                                 blockin + !breakgain]
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   115
                then blanks wd text
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   116
                else blanks blockin (newline text)));
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   117
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   118
(*** Exported functions to create formatting expressions ***)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   119
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   120
fun length (Block(_,_,len)) = len
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   121
  | length (String s) = size s
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   122
  | length (Break(_,wd)) = wd;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   123
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   124
val str = String
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   125
and fbrk = Break(true,0);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   126
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   127
fun brk wd = Break(false,wd);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   128
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   129
fun blk (indent,es) =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   130
  let fun sum([], k) = k
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   131
        | sum(e::es, k) = sum(es, length e + k)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   132
  in  Block(es,indent, sum(es,0))  end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   133
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   134
fun lst(lp,rp) es =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   135
  let fun add(e,es) = str"," :: brk 1 :: e :: es;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   136
      fun list(e :: (es as _::_)) = str lp :: e :: foldr add (es,[str rp])
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   137
        | list(e::[]) = [str lp, e, str rp]
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   138
        | list([]) = []
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   139
  in blk(size lp, list es) end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   140
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   141
fun quote prt = blk (1, [str "\"", prt, str "\""]);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   142
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   143
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   144
(*** Pretty printing with depth limitation ***)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   145
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   146
val depth       = ref 0;        (*maximum depth; 0 means no limit*)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   147
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   148
fun setdepth dp = (depth := dp);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   149
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   150
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   151
fun pruning dp (Block(bes,indent,wd)) =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   152
      if dp>0 then blk(indent, map (pruning(dp-1)) bes)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   153
              else String"..."
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   154
  | pruning dp e = e;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   155
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   156
fun prune dp e = if dp>0 then pruning dp e else e;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   157
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   158
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   159
fun string_of e = #tx (format ([prune (!depth) e],0,0) emptytext);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   160
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   161
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   162
fun brk_width (force, wd) = if force andalso wd = 0 then 1 else wd;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   163
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   164
fun str_of prt =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   165
  let
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   166
    fun s_of (Block (prts, _, _)) = implode (map s_of prts)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   167
      | s_of (String s) = s
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   168
      | s_of (Break f_w) = repstring " " (brk_width f_w);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   169
  in
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   170
    s_of (prune (! depth) prt)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   171
  end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   172
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   173
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   174
fun pprint prt (put_str, begin_blk, put_brk, end_blk) =
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   175
  let
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   176
    fun pp (Block (prts, ind, _)) = (begin_blk ind; pp_lst prts; end_blk ())
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   177
      | pp (String s) = put_str s
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   178
      | pp (Break f_w) = put_brk (brk_width f_w)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   179
    and pp_lst [] = ()
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   180
      | pp_lst (prt :: prts) = (pp prt; pp_lst prts);
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   181
  in
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   182
    pp (prune (! depth) prt)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   183
  end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   184
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   185
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   186
(*** Initialization ***)
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   187
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   188
val () = setmargin 80;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   189
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   190
end;
a5a9c433f639 Initial revision
clasohm
parents:
diff changeset
   191