%% %% %CopyrightBegin% %% %% Copyright Ericsson AB 1998-2009. All Rights Reserved. %% %% The contents of this file are subject to the Erlang Public License, %% Version 1.1, (the "License"); you may not use this file except in %% compliance with the License. You should have received a copy of the %% Erlang Public License along with this software. If not, it can be %% retrieved online at http://www.erlang.org/. %% %% Software distributed under the License is distributed on an "AS IS" %% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See %% the License for the specific language governing rights and limitations %% under the License. %% %% %CopyrightEnd% -module(tv_io_lib_pretty). -export([pretty_print/4]). %% pretty_print(Term, Column, LineLength, Depth) -> [Chars] %% Depth = -1 gives unlimited print depth. Use tv_io_lib:write for atomic terms. pretty_print(_, _, _, 0) -> "..."; pretty_print([], _, _, _) -> "[]"; pretty_print({}, _, _, _) -> "{}"; pretty_print(List, Col, Ll, D) when is_list(List) -> case tv_io_lib:printable_list(List) of true -> tv_io_lib:write_string(List, $"); false -> Len = write_length(List, D, 0, Ll - Col), if D =:= 1 -> "[...]"; Len + Col < Ll -> write(List, D); true -> [$[, [pretty_print(hd(List), Col + 1, Ll, D - 1)| pretty_print_tail(tl(List), Col + 1, Ll, D - 1)], $]] end end; pretty_print(Fun, _Col, _Ll, _D) when is_function(Fun) -> tv_io_lib:write(Fun); pretty_print(Tuple, Col, Ll, D) when is_tuple(Tuple) -> Len = write_length(Tuple, D, 0, Ll - Col), if D =:= 1 -> "{...}"; Len + Col < Ll -> write(Tuple, D); is_atom(element(1, Tuple)), size(Tuple) > 1 -> print_tag_tuple(Tuple, Col, Ll, D); true -> [${, [pretty_print(element(1, Tuple), Col + 1, Ll, D - 1)| pretty_print_tail(tl(tuple_to_list(Tuple)), Col + 1, Ll, D - 1)], $}] end; pretty_print(Term, _Col, _Ll, D) -> tv_io_lib:write(Term, D). %% print_tag_tuple(Tuple, Column, LineLength, Depth) -> [Char] %% Print a tagged tuple by indenting the rest of the elements differently %% to the tag. Start beside the tag if start column not too far to %% the right. Tuple has size >= 2. print_tag_tuple(Tuple, Col, Ll, D) -> Tag = tv_io_lib:write_atom(element(1, Tuple)), Tlen = length(Tag), Tcol = Col + Tlen + 2, if Tcol >= Ll div 2, Tlen > 2 -> [${,Tag, pretty_print_tail(tl(tuple_to_list(Tuple)), Col + 4, Ll, D - 2), $}]; true -> [${,Tag,$,, [pretty_print(element(2, Tuple), Col + Tlen + 2, Ll, D - 2)| pretty_print_tail(tl(tl(tuple_to_list(Tuple))), Tcol, Ll, D - 3)], $}] end. %% pretty_print_tail([Element], Column, LineLength, D) -> [Char] %% Pretty print the elements of a list or tuple. pretty_print_tail([], _Col, _Ll, _D) -> ""; pretty_print_tail(_Es, _Col, _Ll, 1) -> "|..."; pretty_print_tail([E|Es], Col, Ll, D) -> [$,,nl_indent(Col-1), pretty_print(E, Col, Ll, D-1)| pretty_print_tail(Es, Col, Ll, D-1)]; pretty_print_tail(E, Col, Ll, D) -> [$|,nl_indent(Col-1),pretty_print(E, Col, Ll, D-1)]. %% write(Term, Depth) -> [Char] %% Write a term down to Depth on one line. Use tv_io_lib:write/2 for %% atomic terms. write(_, 0) -> "..."; write([], _) -> "[]"; write({}, _) -> "{}"; write(List, D) when is_list(List) -> case tv_io_lib:printable_list(List) of true -> tv_io_lib:write_string(List, $"); false -> if D =:= 1 -> "[...]"; true -> [$[, [write(hd(List), D-1)|write_tail(tl(List), D-1)], $]] end end; write(Fun, _D) when is_function(Fun) -> tv_io_lib:write(Fun); %Must catch this first write(T, D) when is_tuple(T) -> if D =:= 1 -> "{...}"; true -> [${, [write(element(1, T), D-1)|write_tail(tl(tuple_to_list(T)), D-1)], $}] end; write(Term, D) -> tv_io_lib:write(Term, D). write_tail([], _D) -> ""; write_tail(_Es, 1) -> "|..."; write_tail([E|Es], D) -> [$,,write(E, D - 1)|write_tail(Es, D - 1)]; write_tail(E, D) -> [$|,write(E, D - 1)]. %% write_length(Term, Depth, Accumulator, MaxLength) -> integer() %% Calculate the print length of a term, but exit when length becomes %% greater than MaxLength. write_length(_T, _D, Acc, Max) when Acc > Max -> Acc; write_length(_T, 0, Acc, _Max) -> Acc + 3; write_length([], _, Acc, _) -> Acc + 2; write_length({}, _, Acc, _) -> Acc + 2; write_length(List, D, Acc, Max) when is_list(List) -> case tv_io_lib:printable_list(List) of true -> Acc + length(tv_io_lib:write_string(List, $")); false -> write_length_list(List, D, Acc, Max) end; write_length(Fun, _D, Acc, _Max) when is_function(Fun) -> Acc + length(tv_io_lib:write(Fun)); write_length(Tuple, D, Acc, Max) when is_tuple(Tuple) -> write_length_list(tuple_to_list(Tuple), D, Acc, Max); write_length(Term, _D, Acc, _Max) -> Acc + length(tv_io_lib:write(Term)). write_length_list(_, _, Acc, Max) when Acc > Max -> Acc; write_length_list([], _, Acc, _) -> Acc + 1; %] write_length_list(_Es, 1, Acc, _) -> Acc + 5; %|...] write_length_list([E|Es], D, Acc, Max) -> write_length_list(Es, D - 1, write_length(E, D - 1, Acc + 1, Max), Max); write_length_list(E, D, Acc, Max) -> write_length(E, D - 1, Acc + 2, Max). %| ] nl_indent(_) -> "".