1
0
mirror of https://github.com/golang/go synced 2024-11-26 07:07:57 -07:00

- filter trailing whitespace

- removed some unused code

R=rsc
DELTA=103  (84 added, 15 deleted, 4 changed)
OCL=34816
CL=34829
This commit is contained in:
Robert Griesemer 2009-09-19 11:52:40 -07:00
parent 084577b81a
commit 144580d690
4 changed files with 95 additions and 26 deletions

View File

@ -6,6 +6,7 @@
package printer package printer
import ( import (
"bytes";
"container/vector"; "container/vector";
"fmt"; "fmt";
"go/ast"; "go/ast";
@ -74,7 +75,6 @@ type printer struct {
// current state (changes during printing) // current state (changes during printing)
written int; // number of bytes written written int; // number of bytes written
level int; // function nesting level; 0 = package scope, 1 = top-level function scope, etc.
indent int; // current indentation indent int; // current indentation
last token.Position; // (possibly estimated) position immediately after the last item; in AST space last token.Position; // (possibly estimated) position immediately after the last item; in AST space
pos token.Position; // (possibly estimated) position; in AST space pos token.Position; // (possibly estimated) position; in AST space
@ -97,8 +97,8 @@ func (p *printer) init(output io.Writer, mode uint) {
} }
// Writing to p.output is done with write0 which also handles errors. // write0 writes raw (uninterpreted) data to p.output and handles errors.
// Does not indent after newlines, or HTML-escape, or update p.pos. // write0 does not indent after newlines, and does not HTML-escape or update p.pos.
// //
func (p *printer) write0(data []byte) { func (p *printer) write0(data []byte) {
n, err := p.output.Write(data); n, err := p.output.Write(data);
@ -109,23 +109,19 @@ func (p *printer) write0(data []byte) {
} }
// write interprets data and writes it to p.output. It inserts indentation
// after newline or formfeed, converts formfeed characters into newlines if
// RawFormat is set, and HTML-escapes data if GenHTML is set.
//
func (p *printer) write(data []byte) { func (p *printer) write(data []byte) {
i0 := 0; i0 := 0;
for i, b := range data { for i, b := range data {
switch b { switch b {
case '\n', '\f': case '\n', '\f':
// write segment ending in b followed by indentation // write segment ending in b followed by indentation
if p.mode & RawFormat != 0 && b == '\f' {
// no tabwriter - convert last byte into a newline
p.write0(data[i0 : i]);
p.write0(newlines[0 : 1]);
} else {
p.write0(data[i0 : i+1]); p.write0(data[i0 : i+1]);
}
// write indentation // write indentation
// TODO(gri) should not write indentation if there is nothing else
// on the line
j := p.indent; j := p.indent;
for ; j > len(tabs); j -= len(tabs) { for ; j > len(tabs); j -= len(tabs) {
p.write0(&tabs); p.write0(&tabs);
@ -532,19 +528,17 @@ func (p *printer) exprList(list []ast.Expr, mode exprListMode) {
func (p *printer) parameters(list []*ast.Field) { func (p *printer) parameters(list []*ast.Field) {
p.print(token.LPAREN); p.print(token.LPAREN);
if len(list) > 0 { if len(list) > 0 {
p.level++; // adjust nesting level for parameters
for i, par := range list { for i, par := range list {
if i > 0 { if i > 0 {
p.print(token.COMMA, blank); p.print(token.COMMA, blank);
} }
p.identList(par.Names); // p.level > 0; all identifiers will be printed p.identList(par.Names);
if len(par.Names) > 0 { if len(par.Names) > 0 {
// at least one identifier // at least one identifier
p.print(blank); p.print(blank);
}; };
p.expr(par.Type); p.expr(par.Type);
} }
p.level--;
} }
p.print(token.RPAREN); p.print(token.RPAREN);
} }
@ -760,9 +754,7 @@ func (p *printer) expr1(expr ast.Expr, prec1 int) (optSemi bool) {
case *ast.FuncLit: case *ast.FuncLit:
p.expr(x.Type); p.expr(x.Type);
p.print(blank); p.print(blank);
p.level++; // adjust nesting level for function body
p.stmt(x.Body); p.stmt(x.Body);
p.level--;
case *ast.ParenExpr: case *ast.ParenExpr:
p.print(token.LPAREN); p.print(token.LPAREN);
@ -1255,9 +1247,7 @@ func (p *printer) decl(decl ast.Decl) (comment *ast.CommentGroup, optSemi bool)
p.signature(d.Type.Params, d.Type.Results); p.signature(d.Type.Params, d.Type.Results);
if d.Body != nil { if d.Body != nil {
p.print(blank); p.print(blank);
p.level++; // adjust nesting level for function body
p.stmt(d.Body); p.stmt(d.Body);
p.level--;
} }
default: default:
@ -1286,6 +1276,79 @@ func (p *printer) file(src *ast.File) {
} }
// ----------------------------------------------------------------------------
// Trimmer
// A trimmer is an io.Writer filter for stripping trailing blanks
// and tabs, and for converting formfeed characters into newlines.
//
type trimmer struct {
output io.Writer;
buf bytes.Buffer;
}
func (p *trimmer) Write(data []byte) (n int, err os.Error) {
// m < 0: no unwritten data except for whitespace
// m >= 0: data[m:n] unwritten and no whitespace
m := 0;
if p.buf.Len() > 0 {
m = -1;
}
var b byte;
for n, b = range data {
switch b {
default:
// write any pending whitespace
if m < 0 {
if _, err = p.output.Write(p.buf.Bytes()); err != nil {
return;
}
p.buf.Reset();
m = n;
}
case '\t', ' ':
// write any pending (non-whitespace) data
if m >= 0 {
if _, err = p.output.Write(data[m:n]); err != nil {
return;
}
m = -1;
}
// collect whitespace
p.buf.WriteByte(b); // WriteByte returns no errors
case '\f', '\n':
// discard whitespace
p.buf.Reset();
// write any pending (non-whitespace) data
if m >= 0 {
if _, err = p.output.Write(data[m:n]); err != nil {
return;
}
m = -1;
}
// convert formfeed into newline
if _, err = p.output.Write(newlines[0:1]); err != nil {
return;
}
}
}
n = len(data);
// write any pending non-whitespace
if m >= 0 {
if _, err = p.output.Write(data[m:n]); err != nil {
return;
}
}
return;
}
// ---------------------------------------------------------------------------- // ----------------------------------------------------------------------------
// Public interface // Public interface
@ -1298,6 +1361,12 @@ var inf = token.Position{Offset: 1<<30, Line: 1<<30}
// is controlled by the mode and tabwidth parameters. // is controlled by the mode and tabwidth parameters.
// //
func Fprint(output io.Writer, node interface{}, mode uint, tabwidth int) (int, os.Error) { func Fprint(output io.Writer, node interface{}, mode uint, tabwidth int) (int, os.Error) {
// redirect output through a trimmer to eliminate trailing whitespace
// (Input to a tabwriter must be untrimmed since trailing tabs provide
// formatting information. The tabwriter could provide trimming
// functionality but no tabwriter is used when RawFormat is set.)
output = &trimmer{output: output};
// setup tabwriter if needed and redirect output // setup tabwriter if needed and redirect output
var tw *tabwriter.Writer; var tw *tabwriter.Writer;
if mode & RawFormat == 0 { if mode & RawFormat == 0 {