more work on elastic tabs:
- new code enabled, but no comments printed yet (so the effect
  of the elastic tabs is not seen yet)

TBR=r
DELTA=200  (93 added, 69 deleted, 38 changed)
OCL=18951
CL=18951
diff --git a/usr/gri/pretty/printer.go b/usr/gri/pretty/printer.go
index 4c886ad..ea670eb 100644
--- a/usr/gri/pretty/printer.go
+++ b/usr/gri/pretty/printer.go
@@ -36,95 +36,116 @@
 // (http://nickgravgaard.com/elastictabstops/index.html)
 
 type Buffer struct {
+	segment string;  // current line segment
 	lines AST.List;  // a list of lines; and each line is a list of strings
-	widths AST.List;
+}
+
+
+func (b *Buffer) Line(i int) *AST.List {
+	return b.lines.at(i).(*AST.List);
+}
+
+
+func (b *Buffer) Tab() {
+	b.lines.at(b.lines.len() - 1).(*AST.List).Add(b.segment);
+	b.segment = "";
 }
 
 
 func (b *Buffer) Newline() {
+	b.Tab();  // add last segment to current line
 	b.lines.Add(AST.NewList());
 }
 
 
+func (b *Buffer) Print(s string) {
+	b.segment += s;
+}
+
+
 func (b *Buffer) Init() {
 	b.lines.Init();
-	b.widths.Init();
-	b.Newline();
+	b.lines.Add(AST.NewList());
 }
 
 
-func (b *Buffer) ComputeWidths() {
-	// iterate through all columns j
-	for j := 0; ; j++ {
-		width := -1;  // initial column width
-		
-		// iterate through all lines i
-		for i := 0; i < b.lines.len(); i++ {
-			line := b.lines.at(i).(*AST.List);
-			if j < line.len() {
-				// the j.th column exists in this line
-				w := len(line.at(j).(string));
-				if w > width {
-					width = w;
+func (b *Buffer) PrintLines(line0, line1 int, widths *AST.List) {
+	for i := line0; i < line1; i++ {
+		nsep := 0;
+		line := b.Line(i);
+		for j := 0; j < line.len(); j++ {
+			s := line.at(j).(string);
+			PrintBlanks(nsep);
+			print(s);
+			if j < widths.len() {
+				nsep = widths.at(j).(int) - len(s);
+				assert(nsep >= 0);
+				if nsep < int(tabwith.IVal()) {
+					nsep = int(tabwith.IVal());
 				}
+			} else {
+				nsep = 0;
 			}
 		}
-	
-		if width >= 0 {
-			assert(b.widths.len() == j);
-			b.widths.Add(width);
+		println();
+	}
+}
+
+
+func (b *Buffer) Format(line0, line1 int, widths *AST.List) {
+	i0, i1 := line0, line0;
+	column := widths.len();
+	width := -1;
+	for i := line0; i < line1; i++ {
+		line := b.Line(i);
+		if column < line.len() - 1 {
+			if width < 0 {
+				// column start
+				i1 = i;
+				b.PrintLines(i0, i1, widths);
+			}
+			w := len(line.at(column).(string));
+			if w > width {
+				width = w;
+			}
 		} else {
-			// no column j - we are done
-			return;
+			if width >= 0 {
+				// column end
+				i0 = i;
+				widths.Add(width);
+				b.Format(i1, i0, widths);
+				widths.Pop();
+				width = -1;
+			}
 		}
 	}
+	b.PrintLines(i0, line1, widths);
+}
+
+
+func (b *Buffer) Dump() {
+	for i := 0; i < b.lines.len(); i++ {
+		line := b.Line(i);
+		print("(", i, ") ");
+		for j := 0; j < line.len(); j++ {
+			print("[", line.at(j).(string), "]");
+		}
+		print("\n");
+	}
+	print("\n");
 }
 
 
 func (b *Buffer) Flush() {
-	b.ComputeWidths();
-	
-	// print the lines
-	for i := 0; i < b.lines.len(); i++ {
-		line := b.lines.at(i).(*AST.List);
-		for j := 0; j < line.len(); j++ {
-			s := line.at(j).(string);
-			d := b.widths.at(j).(int) - len(s);
-			assert(d >= 0);
-			if d < int(tabwith.IVal()) {
-				d = int(tabwith.IVal());
-			}
-			PrintBlanks(d);  // +1 padding
-			print(s);
-		}
-		println();
-	}
-	
+	b.Tab();  // add last segment to current line
+	b.Format(0, b.lines.len(), AST.NewList());
 	b.lines.Clear();
-	b.widths.Clear();
-	b.Newline();
+	b.lines.Add(AST.NewList());
 }
 
 
-func (b *Buffer) Indent(n int) {
-	line := b.lines.at(b.lines.len() - 1).(*AST.List);
-	for ; n > 0; n-- {
-		line.Add("");
-	}
-}
-
-
-func (b *Buffer) Print(s string) {
-	i := b.lines.len() - 1;
-	line := b.lines.at(i).(*AST.List);
-	j := line.len() - 1;
-	if j < 0 {
-		line.Add(s);
-	} else {
-		line.set(j, line.at(j).(string) + s);
-	}
-}
-
+// ----------------------------------------------------------------------------
+// Printer
 
 export type Printer struct {
 	buf Buffer;
@@ -142,28 +163,42 @@
 }
 
 
-const NEW_CODE = false;
+func CountNewlinesAndTabs(s string) (int, int, string) {
+	nls, tabs := 0, 0;
+	for i := 0; i < len(s); i++ {
+		switch ch := s[i]; ch {
+		case '\n': nls++;
+		case '\t': tabs++;
+		case ' ':
+		default:
+			// non-whitespace char
+			assert(ch == '/');
+			return nls, tabs, s[i : len(s)];
+		}
+	}
+	return nls, tabs, "";
+}
+
 
 func (P *Printer) String(pos int, s string) {
 	if P.semi && P.level > 0 {  // no semicolons at level 0
-		if NEW_CODE {
-			P.buf.Print(";");
-		} else {
-			print(";");
-		}
+		P.buf.Print(";");
 	}
 
 	/*
 	for pos > P.cpos {
 		// we have a comment
-		c := P.clist.at(P.cindex).(*AST.Comment);
-		if len(c.text) > 1 && c.text[1] == '/' {
-			print("  " + c.text);
+		comment := P.clist.at(P.cindex).(*AST.Comment);
+		nls, tabs, text := CountNewlinesAndTabs(comment.text);
+		
+		if nls == 0 && len(text) > 1 && text[1] == '/' {
+			P.buf.Tab();
+			P.buf.Print(text);
 			if P.newl <= 0 {
-				P.newl = 1;  // line comments must have a newline
+				//P.newl = 1;  // line comments must have a newline
 			}
 		} else {
-			print(c.text);
+			P.buf.Print(text);
 		}
 		P.cindex++;
 		if P.cindex < P.clist.len() {
@@ -175,30 +210,19 @@
 	*/
 
 	if P.newl > 0 {
-		if NEW_CODE {
-			P.buf.Flush();
-		}
-		for i := P.newl; i > 0; i-- {
-			if NEW_CODE {
+		P.buf.Newline();
+		if P.newl > 1 {
+			for i := P.newl; i > 1; i-- {
+				//P.buf.Flush();
 				P.buf.Newline();
-			} else {
-				print("\n");
 			}
 		}
-		if NEW_CODE {
-			P.buf.Indent(P.indent);
-		} else {
-			for i := P.indent; i > 0; i-- {
-				print("\t");
-			}
+		for i := P.indent; i > 0; i-- {
+			P.buf.Tab();
 		}
 	}
 
-	if NEW_CODE {
-		P.buf.Print(s);
-	} else {
-		print(s);
-	}
+	P.buf.Print(s);
 
 	P.semi, P.newl = false, 0;
 }
@@ -668,5 +692,5 @@
 	}
 	P.newl = 1;
 
-	P.String(0, "");  // flush
+	P.buf.Flush();  // TODO should not access P.buf directly here
 }