aboutsummaryrefslogtreecommitdiff
path: root/formatter/formatter.go
blob: 7d4489150f61877582a4e57310534ce4c1c90c0d (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
package formatter

import (
	"fmt"
	"unicode"

	"git.thomasvoss.com/gsp/parser"
)

var stringEscapes = map[rune]string{
	'"': """,
	'&': "&",
	'<': "&lt;",
}

func PrintHtml(ast parser.AstNode) {
	if ast.Type == parser.Text {
		fmt.Print(ast.Text)
		return
	}

	if ast.Type == parser.Normal {
		fmt.Printf("<%s", ast.Text)

		// Classes are grouped together with ‘class="…"’, so we need
		// special handling.
		classes := []string{}
		notClasses := []parser.Attr{}

		for _, a := range ast.Attrs {
			if a.Key == "class" {
				classes = append(classes, a.Value)
			} else {
				notClasses = append(notClasses, a)
			}
		}

		if len(classes) > 0 {
			fmt.Printf(" class=\"%s", classes[0])
			for _, c := range classes[1:] {
				fmt.Printf(" %s", c)
			}
			fmt.Print("\"")
		}

		for _, a := range notClasses {
			fmt.Printf(" %s", a.Key)
			if a.Value == "" {
				continue
			}
			fmt.Print("=\"")
			for _, r := range a.Value {
				if v, ok := stringEscapes[r]; ok {
					fmt.Print(v)
				} else {
					fmt.Printf("%c", r)
				}
			}
			fmt.Print("\"")
		}

		fmt.Print(">")
	}

	if len(ast.Children) == 0 {
		return
	}

	for i, n := range ast.Children {
		if n.Type == parser.Text {
			if i == 0 {
				n.Text = trimLeftSpaces(n.Text)
			}

			if i == len(ast.Children)-1 {
				n.Text = trimRightSpaces(n.Text)
			}
		}

		PrintHtml(n)
	}

	if ast.Type == parser.Normal {
		fmt.Printf("</%s>", ast.Text)
	}
}

func trimLeftSpaces(s string) string {
	i := 0
	rs := []rune(s)
	for i < len(s) && unicode.IsSpace(rs[i]) {
		i++
	}
	return string(rs[i:])
}

func trimRightSpaces(s string) string {
	i := len(s) - 1
	rs := []rune(s)
	for i >= 0 && unicode.IsSpace(rs[i]) {
		i--
	}
	return string(rs[:i+1])
}