1
0
mirror of https://github.com/astaxie/beego.git synced 2024-11-04 22:00:54 +00:00
Beego/template.go

207 lines
4.6 KiB
Go
Raw Normal View History

2013-03-14 09:57:09 +00:00
package beego
//@todo add template funcs
import (
"errors"
"fmt"
"html/template"
2013-09-12 09:20:32 +00:00
"io/ioutil"
2013-03-14 09:57:09 +00:00
"os"
"path/filepath"
2013-09-12 09:20:32 +00:00
"regexp"
2013-03-14 09:57:09 +00:00
"strings"
)
var (
2013-09-12 09:20:32 +00:00
beegoTplFuncMap template.FuncMap
BeeTemplates map[string]*template.Template
BeeTemplateExt []string
2013-03-14 09:57:09 +00:00
)
func init() {
BeeTemplates = make(map[string]*template.Template)
beegoTplFuncMap = make(template.FuncMap)
BeeTemplateExt = make([]string, 0)
BeeTemplateExt = append(BeeTemplateExt, "tpl", "html")
beegoTplFuncMap["dateformat"] = DateFormat
beegoTplFuncMap["date"] = Date
beegoTplFuncMap["compare"] = Compare
2013-03-14 13:47:39 +00:00
beegoTplFuncMap["substr"] = Substr
2013-03-21 05:21:04 +00:00
beegoTplFuncMap["html2str"] = Html2str
beegoTplFuncMap["str2html"] = Str2html
beegoTplFuncMap["htmlquote"] = Htmlquote
beegoTplFuncMap["htmlunquote"] = Htmlunquote
2013-08-10 08:33:46 +00:00
beegoTplFuncMap["renderform"] = RenderForm
2013-03-14 09:57:09 +00:00
}
// AddFuncMap let user to register a func in the template
func AddFuncMap(key string, funname interface{}) error {
if _, ok := beegoTplFuncMap[key]; ok {
return errors.New("funcmap already has the key")
}
beegoTplFuncMap[key] = funname
return nil
}
type templatefile struct {
root string
files map[string][]string
}
func (self *templatefile) visit(paths string, f os.FileInfo, err error) error {
if f == nil {
return err
}
2013-08-01 03:57:29 +00:00
if f.IsDir() || (f.Mode()&os.ModeSymlink) > 0 {
2013-03-14 09:57:09 +00:00
return nil
2013-08-01 03:57:29 +00:00
}
if !HasTemplateEXt(paths) {
2013-03-14 09:57:09 +00:00
return nil
2013-08-01 03:57:29 +00:00
}
replace := strings.NewReplacer("\\", "/")
a := []byte(paths)
a = a[len([]byte(self.root)):]
2013-09-12 09:20:32 +00:00
file := strings.TrimLeft(replace.Replace(string(a)), "/")
subdir := filepath.Dir(file)
2013-08-01 03:57:29 +00:00
if _, ok := self.files[subdir]; ok {
2013-09-13 05:57:40 +00:00
self.files[subdir] = append(self.files[subdir], file)
2013-03-14 09:57:09 +00:00
} else {
2013-08-01 03:57:29 +00:00
m := make([]string, 1)
2013-09-13 05:57:40 +00:00
m[0] = file
2013-08-01 03:57:29 +00:00
self.files[subdir] = m
}
2013-03-14 09:57:09 +00:00
2013-08-01 03:57:29 +00:00
return nil
}
func HasTemplateEXt(paths string) bool {
for _, v := range BeeTemplateExt {
if strings.HasSuffix(paths, "."+v) {
return true
2013-03-14 09:57:09 +00:00
}
}
2013-08-01 03:57:29 +00:00
return false
2013-03-14 09:57:09 +00:00
}
func AddTemplateExt(ext string) {
for _, v := range BeeTemplateExt {
if v == ext {
return
}
}
BeeTemplateExt = append(BeeTemplateExt, ext)
}
func BuildTemplate(dir string) error {
2013-03-21 13:45:48 +00:00
if _, err := os.Stat(dir); err != nil {
if os.IsNotExist(err) {
return nil
2013-03-21 13:45:48 +00:00
} else {
return errors.New("dir open err")
}
2013-03-21 12:46:54 +00:00
}
2013-09-12 09:20:32 +00:00
self := &templatefile{
2013-03-14 09:57:09 +00:00
root: dir,
files: make(map[string][]string),
}
2013-03-21 13:47:19 +00:00
err := filepath.Walk(dir, func(path string, f os.FileInfo, err error) error {
2013-09-12 09:20:32 +00:00
return self.visit(path, f, err)
2013-03-14 09:57:09 +00:00
})
if err != nil {
fmt.Printf("filepath.Walk() returned %v\n", err)
return err
}
2013-09-13 05:57:40 +00:00
for _, v := range self.files {
for _, file := range v {
t, err := getTemplate(self.root, file, v...)
if err != nil {
Trace("parse template err:", file, err)
} else {
BeeTemplates[file] = t
}
}
}
2013-03-14 09:57:09 +00:00
return nil
}
2013-09-12 09:20:32 +00:00
2013-09-13 09:41:31 +00:00
func getTplDeep(root, file string, t *template.Template) (*template.Template, [][]string, error) {
2013-09-12 09:32:11 +00:00
fileabspath := filepath.Join(root, file)
2013-09-12 09:20:32 +00:00
data, err := ioutil.ReadFile(fileabspath)
if err != nil {
2013-09-13 09:41:31 +00:00
return nil, [][]string{}, err
2013-09-12 09:20:32 +00:00
}
t, err = t.New(file).Parse(string(data))
if err != nil {
2013-09-13 09:41:31 +00:00
return nil, [][]string{}, err
2013-09-12 09:20:32 +00:00
}
2013-09-12 10:20:29 +00:00
reg := regexp.MustCompile("{{[ ]*template[ ]+\"([^\"]+)\"")
2013-09-12 09:20:32 +00:00
allsub := reg.FindAllStringSubmatch(string(data), -1)
for _, m := range allsub {
if len(m) == 2 {
tlook := t.Lookup(m[1])
if tlook != nil {
continue
}
if !HasTemplateEXt(m[1]) {
continue
}
2013-09-13 09:41:31 +00:00
t, _, err = getTplDeep(root, m[1], t)
2013-09-12 09:20:32 +00:00
if err != nil {
2013-09-13 09:41:31 +00:00
return nil, [][]string{}, err
2013-09-12 09:20:32 +00:00
}
}
}
2013-09-13 09:41:31 +00:00
return t, allsub, nil
2013-09-12 09:20:32 +00:00
}
2013-09-13 05:57:40 +00:00
func getTemplate(root, file string, others ...string) (t *template.Template, err error) {
2013-09-12 09:20:32 +00:00
t = template.New(file).Delims(TemplateLeft, TemplateRight).Funcs(beegoTplFuncMap)
2013-09-13 09:41:31 +00:00
var submods [][]string
t, submods, err = getTplDeep(root, file, t)
for _, m := range submods {
if len(m) == 2 {
templ := t.Lookup(m[1])
if templ != nil {
continue
}
//first check filename
for _, otherfile := range others {
if otherfile == m[1] {
t, _, err = getTplDeep(root, otherfile, t)
if err != nil {
Trace("template parse file err:", err)
}
break
}
}
//second check define
for _, otherfile := range others {
fileabspath := filepath.Join(root, otherfile)
data, err := ioutil.ReadFile(fileabspath)
if err != nil {
continue
}
reg := regexp.MustCompile("{{[ ]*define[ ]+\"([^\"]+)\"")
allsub := reg.FindAllStringSubmatch(string(data), -1)
for _, sub := range allsub {
if len(sub) == 2 && sub[1] == m[1] {
t, _, err = getTplDeep(root, otherfile, t)
if err != nil {
Trace("template parse file err:", err)
}
break
}
}
}
2013-09-13 05:57:40 +00:00
}
2013-09-13 09:41:31 +00:00
2013-09-13 05:57:40 +00:00
}
2013-09-13 09:41:31 +00:00
2013-09-12 09:20:32 +00:00
if err != nil {
return nil, err
}
return
}