aboutsummaryrefslogtreecommitdiff
path: root/libgo/go/html/template/clone.go
blob: d0d8ea46733f4aed2538603f619fec906c275f21 (plain)
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
// Copyright 2011 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

package template

import (
	"text/template/parse"
)

// clone clones a template Node.
func clone(n parse.Node) parse.Node {
	switch t := n.(type) {
	case *parse.ActionNode:
		return cloneAction(t)
	case *parse.IfNode:
		b := new(parse.IfNode)
		copyBranch(&b.BranchNode, &t.BranchNode)
		return b
	case *parse.ListNode:
		return cloneList(t)
	case *parse.RangeNode:
		b := new(parse.RangeNode)
		copyBranch(&b.BranchNode, &t.BranchNode)
		return b
	case *parse.TemplateNode:
		return cloneTemplate(t)
	case *parse.TextNode:
		return cloneText(t)
	case *parse.WithNode:
		b := new(parse.WithNode)
		copyBranch(&b.BranchNode, &t.BranchNode)
		return b
	}
	panic("cloning " + n.String() + " is unimplemented")
}

// cloneAction returns a deep clone of n.
func cloneAction(n *parse.ActionNode) *parse.ActionNode {
	// We use keyless fields because they won't compile if a field is added.
	return &parse.ActionNode{n.NodeType, n.Line, clonePipe(n.Pipe)}
}

// cloneList returns a deep clone of n.
func cloneList(n *parse.ListNode) *parse.ListNode {
	if n == nil {
		return nil
	}
	// We use keyless fields because they won't compile if a field is added.
	c := parse.ListNode{n.NodeType, make([]parse.Node, len(n.Nodes))}
	for i, child := range n.Nodes {
		c.Nodes[i] = clone(child)
	}
	return &c
}

// clonePipe returns a shallow clone of n.
// The escaper does not modify pipe descendants in place so there's no need to
// clone deeply.
func clonePipe(n *parse.PipeNode) *parse.PipeNode {
	if n == nil {
		return nil
	}
	// We use keyless fields because they won't compile if a field is added.
	return &parse.PipeNode{n.NodeType, n.Line, n.Decl, n.Cmds}
}

// cloneTemplate returns a deep clone of n.
func cloneTemplate(n *parse.TemplateNode) *parse.TemplateNode {
	// We use keyless fields because they won't compile if a field is added.
	return &parse.TemplateNode{n.NodeType, n.Line, n.Name, clonePipe(n.Pipe)}
}

// cloneText clones the given node sharing its []byte.
func cloneText(n *parse.TextNode) *parse.TextNode {
	// We use keyless fields because they won't compile if a field is added.
	return &parse.TextNode{n.NodeType, n.Text}
}

// copyBranch clones src into dst.
func copyBranch(dst, src *parse.BranchNode) {
	// We use keyless fields because they won't compile if a field is added.
	*dst = parse.BranchNode{
		src.NodeType,
		src.Line,
		clonePipe(src.Pipe),
		cloneList(src.List),
		cloneList(src.ElseList),
	}
}