Skip to content
Snippets Groups Projects
pretty_utils.mli 10.46 KiB
(**************************************************************************)
(*                                                                        *)
(*  This file is part of Frama-C.                                         *)
(*                                                                        *)
(*  Copyright (C) 2007-2025                                               *)
(*    CEA (Commissariat à l'énergie atomique et aux énergies              *)
(*         alternatives)                                                  *)
(*                                                                        *)
(*  you can redistribute it and/or modify it under the terms of the GNU   *)
(*  Lesser General Public License as published by the Free Software       *)
(*  Foundation, version 2.1.                                              *)
(*                                                                        *)
(*  It 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 Lesser General Public License for more details.                   *)
(*                                                                        *)
(*  See the GNU Lesser General Public License version 2.1                 *)
(*  for more details (enclosed in the file licenses/LGPLv2.1).            *)
(*                                                                        *)
(**************************************************************************)

(** Pretty-printer utilities.
    @see <https://frama-c.com/download/frama-c-plugin-development-guide.pdf> *)

(* ********************************************************************** *)
(** {2 null formatters} *)
(* ********************************************************************** *)

val null : Format.formatter
(** Prints nothing.
    @since Beryllium-20090901 *)

val nullprintf :  ('a,Format.formatter,unit) format -> 'a
(** Discards the message and returns unit.
    @since Beryllium-20090901 *)

val with_null : (unit -> 'b) -> ('a,Format.formatter,unit,'b) format4 -> 'a
(** Discards the message and call the continuation.
    @since Beryllium-20090901 *)

(* ********************************************************************** *)
(** {2 pretty-printing to a string} *)
(* ********************************************************************** *)

val ksfprintf:
  (string -> 'b) -> ('a, Format.formatter, unit, 'b) format4 -> 'a
(** similar to Format.kfprintf, but the continuation is given the result
    string instead of a formatter.
    @since Magnesium-20151001
*)

val to_string: ?margin:int -> (Format.formatter -> 'a -> unit) -> 'a -> string
(** pretty-prints the supplied value into a string. [margin] is the
    maximal width of the box before a line-break is inserted.
    See {!Format.set_margin} *)

(** {2 separators} *)

val escape_underscores : string -> string

(* ********************************************************************** *)
(** {2 pretty printers for standard types} *)
(* ********************************************************************** *)

type sformat = (unit,Format.formatter,unit) format
type 'a formatter = Format.formatter -> 'a -> unit
type ('a,'b) formatter2 = Format.formatter -> 'a -> 'b -> unit

val pp_list: ?pre:sformat -> ?sep:sformat -> ?last:sformat -> ?suf:sformat ->
  ?empty:sformat -> 'a formatter -> 'a list formatter
(** pretty prints a list. The optional arguments stands for
    - the prefix to output before a non-empty list (default: open a box)
    - the separator between two elements (default: nothing)
    - the last separator to be put just before the last element (default:sep)
    - the suffix to output after a non-empty list (default: close box)
    - what to print if the list is empty (default: nothing)
*)

val pp_array: ?pre:sformat -> ?sep:sformat -> ?suf:sformat -> ?empty:sformat ->
  (int,'a) formatter2 -> 'a array formatter
(** pretty prints an array. The optional arguments stands for
    - the prefix to output before a non-empty array (default: open a box)
    - the separator between two elements (default: nothing)
    - the suffix to output after a non-empty array (default: close box)
    - what to print if the array is empty (default: nothing)
*)

val pp_iter:
  ?pre:sformat -> ?sep:sformat -> ?suf:sformat ->
  (('a -> unit) -> 'b -> unit) ->
  'a formatter -> 'b formatter
(** pretty prints any structure using an iterator on it. The argument
    [pre] (resp. [suf]) is output before (resp. after) the iterator
    is started (resp. has ended). The optional argument [sep] is output between
    two calls to the ['a formatter]. Default: open a box for [pre], close
    a box for [suf], nothing for [sep]. *)

val pp_iter2:
  ?pre:sformat -> ?sep:sformat -> ?suf:sformat -> ?between:sformat ->
  (('key -> 'v -> unit) -> 'a -> unit) ->
  'key formatter -> 'v formatter -> 'a formatter
(** pretty prints any map-like structure using an iterator on it. The argument
    [pre] (resp. [suf]) is output before (resp. after) the iterator
    is started (resp. has ended). The optional argument [sep] is output between
    two calls to the ['a formatter]. The optional argument [between] is
    output between the key and the value. Default: open a box for [pre], close
    a box for [suf], nothing for [sep], break-space for [between]. *)

val pp_opt: ?pre:sformat -> ?suf:sformat -> ?none:sformat -> 'a formatter -> 'a option formatter
(** pretty-prints an optional value. Prefix and suffix default to "@[" and "@]"
    respectively. If the value is [None], pretty-print using [none].
*)

val pp_cond: ?pr_false:sformat -> bool -> sformat formatter
(** [pp_cond cond f s]  pretty-prints [s] if cond is [true] and the optional
    pr_false, which defaults to nothing, otherwise *)

val pp_pair: ?pre:sformat -> ?sep:sformat -> ?suf:sformat ->
  'a formatter -> 'b formatter -> ('a * 'b)  formatter
(** [pp_pair ?pre ?sep ?suf pp_a pp_b (a,b)] pretty prints the pair [(a,b)],
    using the pretty printers [pp_a] and [pp_b], with optional
    prefix/separator/suffix, whose default values are:
    - pre: open a box
    - sep: print a comma character
    - suf: close a box.
      @since Magnesium-20151001 *)

val pp_flowlist:
  ?left:sformat -> ?sep:sformat -> ?right:sformat -> 'a formatter ->
  'a list formatter

val pp_blocklist:
  ?left:sformat -> ?right:sformat -> 'a formatter -> 'a list formatter

val pp_open_block : Format.formatter -> ('a,Format.formatter,unit) format -> 'a
val pp_close_block : Format.formatter -> ('a,Format.formatter,unit) format -> 'a

val pp_trail : 'a formatter -> 'a formatter
(** pretty-prints its contents inside an '(** ... **)' horizontal block trailed
    with '*' *)

val pp_escaped: 'a formatter -> 'a formatter
(** [pp_escaped pp e] pretty-prints [e] with [String.escaped].
    @since 30.0-Zinc *)

(* ********************************************************************** *)
(** {2 Description Lists (margins)} *)
(* ********************************************************************** *)

type align = [`Center | `Left | `Right]

val pp_items :
  ?align:align -> ?margin:int -> ?min:int -> ?max:int ->
  title:('a -> string) ->
  iter:(('a -> unit) -> unit) ->
  ?pp_title:string formatter ->
  pp_item:(string formatter -> 'a formatter) ->
  Format.formatter -> unit
(** Prints a collection of elements, with the possibility of aligning {i titles}
    with each others.

    The collection of ['a] to print is provided by iterator [~iter] which
    is called twice: one for computing the maximal size of {i titles},
    obtained {i via} function [~title] for each item. The second pass
    pretty-print each item using [~pp_item pp] where the passed [pp] printer
    can be used to pretty-print titles with alignment.

    A typical usage for printing [values], a list of [(string*int)] items:
    {[
      pp_items
        ~title:(fun (a,_) -> a)
        ~iter:(fun f -> List.iter f values)
        ~pp_title:(fun fmt a -> Format.fprintf fmt "%s:" a)
        ~pp_item:(fun pp fmt (a,n) -> Format.fprintf fmt "%a %d@\n" pp a n)
        fmt
    ]}

    Alignment of titles can be centered, right or left justified. This is
    rendered by adding spaces around each title. A min and max size can also be
    specified and a margin can be added to all title sizes. Titles will be
    truncated if necessary.

    The pretty-printer for titles will render each (possibly truncated) title
    with [~pp_title]. Surrounding spaces are {i not} printed via [~pp_title].

    The (optional) parameters have the following meaning:
    - [?align] alignment mode (default is [`Center])
    - [?margin] is added to text size (default [0])
    - [?min] minimum size ([~margin] included, default [0])
    - [?max] maximum size ([~margin] included, default [80])
    - [~title] returns the {i title} for each element (only size is relevant)
    - [~iter] iterate over the elements to be printed
    - [?pp_title] pretty-printer used to the (possibly truncated) title
       (default is [Format.pp_print_string])
    - [~pp_item] pretty-printer to print each element.

    There is also a low-level API to this feature, provided by {!marger},
    {!pp_margin} and {!add_margin} below.
*)

(** Margin accumulator (low-level API to [pp_items]). *)
type marger

val marger : unit -> marger
(** Create an empty marger *)

val add_margin : marger -> ?margin:int -> ?min:int -> ?max:int -> string -> unit
(** Updates the marger with new text dimension.
    The marger width is updated with the width of the provided text.
    The optional parameters are used to adjust the text width as follows:
    - [?margin] is added to text size (default [0])
    - [?min] minimum size ([~margin] included, default [0])
    - [?max] maximum size ([~margin] included, default [80]) *)

val pp_margin : ?align:align -> ?pp:string formatter -> marger -> string formatter
(** Prints a text with margins {i wrt} to marger. If the text does not fit
    the marger, it would be truncated and/or ellipsed.
    - [?align] alignment mode (default is [`Center])
    - [?pp] pretty-printer used to the (possibly truncated) title
       (default is [Format.pp_print_string])

    Typical usage:
    {[
      begin
        (* first, collect title margins *)
        let m = marger () in
        List.iter (fun (a,_) -> add_margin m ~margins:2 a) data ;
        (* second, print aligned data *)
        List.iter
          (fun (a,d) -> Format.printf "[%a] %s@\n" (pp_margin m) a d)
          data ;
      end
    ]}
*)

(*
Local Variables:
compile-command: "make -C ../../.."
End:
*)