cleanup and improve formatting logic
[scpubgit/Q-Branch.git] / lib / SQL / Abstract / Formatter.pm
CommitLineData
b3b54441 1package SQL::Abstract::Formatter;
2
3require SQL::Abstract::Parts; # it loads us too, don't cross the streams
4
5use Moo;
6
7has indent_by => (is => 'ro', default => ' ');
8has max_width => (is => 'ro', default => 78);
9
10sub _join {
11 shift;
4ebfbfc1 12 return SQL::Abstract::Parts::stringify(\@_);
b3b54441 13}
14
15sub format {
16 my ($self, $join, @parts) = @_;
07070f1a 17 $self->_fold_sql('', '', @{$self->_simplify($join, @parts)});
18}
19
20sub _simplify {
21 my ($self, $join, @parts) = @_;
22 return '' unless @parts;
23 return $parts[0] if @parts == 1 and !ref($parts[0]);
24 return $self->_simplify(@{$parts[0]}) if @parts == 1;
25 return [ $join, map ref() ? $self->_simplify(@$_) : $_, @parts ];
26}
27
28sub _fold_sql {
07070f1a 29 my ($self, $indent0, $indent, $join, @parts) = @_;
30 my @res;
31 my $w = $self->max_width;
32 my $join_len = 0;
33 (s/, \Z/,\n/ and $join_len = 1)
34 or s/\A /\n/
35 or $_ = "\n"
36 for my $line_join = $join;
37 my ($nl_pre, $nl_post) = split "\n", $line_join;
12fceb64 38 my $line_orig = my $line = $indent0;
07070f1a 39 my $next_indent = $indent.$self->indent_by;
12fceb64 40 my $line_proto = $indent.$nl_post;
07070f1a 41 PART: foreach my $idx (0..$#parts) {
07070f1a 42 my $p = $parts[$idx];
12fceb64 43 my $pre = ($line ne $line_orig ? $join : '');
07070f1a 44 my $j_part = $pre.(my $j = ref($p) ? $self->_join(@$p) : $p);
45 if (length($j_part) + length($line) + $join_len <= $w) {
46 $line .= $j_part;
47 } else {
1f2d8ba1 48 if (ref($p) and $p->[1] eq '(' and $p->[-1] eq ')') {
12fceb64 49 my $already = !($line eq $indent0 or $line eq $line_orig);
50 push @res, $line.($already ? $join : '').'('."\n";
4ebfbfc1 51 my (undef, undef, $inner) = @$p;
12fceb64 52 my $fold_indent = $already ? $indent : $next_indent;
53 my $folded = $self->_fold_sql($fold_indent, $fold_indent, @$inner);
161d8105 54 push @res, $folded."\n";
12fceb64 55 $line_orig = $line
56 = $indent0.')'.($nl_post and $idx < $#parts ? ' '.$nl_post : '');
4ebfbfc1 57 next PART;
58 }
12fceb64 59 push @res, $line.$nl_pre."\n" if $line ne $line_orig;
60 if (length($line = $line_proto.$j) <= $w) {
61 $line_proto = $line;
07070f1a 62 next PART;
63 }
12fceb64 64 my $folded = $self->_fold_sql($line_proto, $next_indent, @$p);
4a9ad1af 65 $folded =~ s/\n\Z//;
e7d3a4e5 66 push @res, $folded.$nl_pre."\n";
12fceb64 67 $line_orig = $line = $idx == $#parts ? '' : $line_proto;
07070f1a 68 }
07070f1a 69 }
4ebfbfc1 70 return join '', @res, $line;
b3b54441 71}
72
731;