2016-10-18 20:30:10 +00:00
|
|
|
// Package page is used to render the HTML to the end user
|
|
|
|
package page
|
2016-06-10 21:18:44 +00:00
|
|
|
|
|
|
|
import (
|
2016-06-22 20:21:32 +00:00
|
|
|
"bytes"
|
2016-12-29 22:50:36 +00:00
|
|
|
"encoding/base64"
|
2016-06-11 09:37:36 +00:00
|
|
|
"encoding/json"
|
2016-06-11 21:15:42 +00:00
|
|
|
"html/template"
|
2016-06-10 21:38:52 +00:00
|
|
|
"log"
|
2016-06-11 09:37:36 +00:00
|
|
|
"net/http"
|
2016-06-11 09:08:33 +00:00
|
|
|
"strings"
|
2016-06-23 22:21:44 +00:00
|
|
|
|
2016-06-28 09:24:02 +00:00
|
|
|
"github.com/hacdias/caddy-filemanager/assets"
|
|
|
|
"github.com/hacdias/caddy-filemanager/config"
|
2016-10-22 11:07:19 +00:00
|
|
|
"github.com/hacdias/caddy-filemanager/utils/variables"
|
2016-06-10 21:18:44 +00:00
|
|
|
)
|
|
|
|
|
2016-10-18 20:30:10 +00:00
|
|
|
// Page contains the informations and functions needed to show the Page
|
|
|
|
type Page struct {
|
|
|
|
*Info
|
2016-06-30 21:37:52 +00:00
|
|
|
Minimal bool
|
2016-06-11 21:15:42 +00:00
|
|
|
}
|
|
|
|
|
2016-10-18 20:30:10 +00:00
|
|
|
// Info contains the information of a Page
|
|
|
|
type Info struct {
|
2016-12-30 22:57:14 +00:00
|
|
|
Name string
|
|
|
|
Path string
|
|
|
|
IsDir bool
|
|
|
|
User *config.User
|
|
|
|
Config *config.Config
|
|
|
|
Data interface{}
|
2016-12-31 15:29:36 +00:00
|
|
|
Editor bool
|
2016-12-30 22:57:14 +00:00
|
|
|
Display string
|
|
|
|
Token string
|
2016-06-11 09:37:36 +00:00
|
|
|
}
|
|
|
|
|
2017-01-14 16:36:40 +00:00
|
|
|
// Commit returns the current build commit
|
|
|
|
func (i *Info) Commit() string {
|
|
|
|
return GIT_COMMIT_HASH
|
|
|
|
}
|
|
|
|
|
2016-12-30 16:22:26 +00:00
|
|
|
// BreadcrumbMapItem ...
|
|
|
|
type BreadcrumbMapItem struct {
|
|
|
|
Name string
|
|
|
|
URL string
|
|
|
|
}
|
|
|
|
|
2016-06-11 09:37:36 +00:00
|
|
|
// BreadcrumbMap returns p.Path where every element is a map
|
|
|
|
// of URLs and path segment names.
|
2016-12-30 16:22:26 +00:00
|
|
|
func (i Info) BreadcrumbMap() []BreadcrumbMapItem {
|
|
|
|
result := []BreadcrumbMapItem{}
|
2016-06-11 09:37:36 +00:00
|
|
|
|
2016-06-25 20:57:10 +00:00
|
|
|
if len(i.Path) == 0 {
|
2016-06-11 09:37:36 +00:00
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip trailing slash
|
2016-06-25 20:57:10 +00:00
|
|
|
lpath := i.Path
|
2016-06-11 09:37:36 +00:00
|
|
|
if lpath[len(lpath)-1] == '/' {
|
|
|
|
lpath = lpath[:len(lpath)-1]
|
|
|
|
}
|
|
|
|
|
|
|
|
parts := strings.Split(lpath, "/")
|
|
|
|
for i, part := range parts {
|
2016-08-26 20:59:06 +00:00
|
|
|
if i == len(parts)-1 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2016-06-11 09:37:36 +00:00
|
|
|
if i == 0 && part == "" {
|
2016-12-30 16:22:26 +00:00
|
|
|
result = append([]BreadcrumbMapItem{{
|
|
|
|
Name: "/",
|
|
|
|
URL: "/",
|
|
|
|
}}, result...)
|
2016-06-11 09:37:36 +00:00
|
|
|
continue
|
|
|
|
}
|
2016-12-30 16:22:26 +00:00
|
|
|
|
|
|
|
result = append([]BreadcrumbMapItem{{
|
|
|
|
Name: part,
|
|
|
|
URL: strings.Join(parts[:i+1], "/"),
|
|
|
|
}}, result...)
|
2016-06-11 09:37:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
2016-06-10 21:18:44 +00:00
|
|
|
}
|
|
|
|
|
2016-06-14 17:19:10 +00:00
|
|
|
// PreviousLink returns the path of the previous folder
|
2016-10-18 20:30:10 +00:00
|
|
|
func (i Info) PreviousLink() string {
|
2016-06-26 15:52:15 +00:00
|
|
|
path := strings.TrimSuffix(i.Path, "/")
|
|
|
|
path = strings.TrimPrefix(path, "/")
|
2016-11-14 20:09:30 +00:00
|
|
|
path = i.Config.AbsoluteURL() + "/" + path
|
2016-06-26 15:52:15 +00:00
|
|
|
path = path[0 : len(path)-len(i.Name)]
|
2016-06-14 17:19:10 +00:00
|
|
|
|
2016-11-14 20:09:30 +00:00
|
|
|
if len(path) < len(i.Config.AbsoluteURL()+"/") {
|
2016-06-26 15:52:15 +00:00
|
|
|
return ""
|
2016-06-14 19:33:59 +00:00
|
|
|
}
|
|
|
|
|
2016-06-26 15:52:15 +00:00
|
|
|
return path
|
2016-06-14 17:19:10 +00:00
|
|
|
}
|
|
|
|
|
2016-06-23 21:27:13 +00:00
|
|
|
// PrintAsHTML formats the page in HTML and executes the template
|
2016-10-18 20:30:10 +00:00
|
|
|
func (p Page) PrintAsHTML(w http.ResponseWriter, templates ...string) (int, error) {
|
2016-06-23 22:21:44 +00:00
|
|
|
// Create the functions map, then the template, check for erros and
|
|
|
|
// execute the template if there aren't errors
|
|
|
|
functions := template.FuncMap{
|
2016-10-22 11:07:19 +00:00
|
|
|
"Defined": variables.Defined,
|
2016-08-04 20:49:49 +00:00
|
|
|
"CSS": func(s string) template.CSS {
|
|
|
|
return template.CSS(s)
|
2016-08-03 14:31:43 +00:00
|
|
|
},
|
2016-08-21 19:44:28 +00:00
|
|
|
"Marshal": func(v interface{}) template.JS {
|
|
|
|
a, _ := json.Marshal(v)
|
|
|
|
return template.JS(a)
|
|
|
|
},
|
2016-12-29 22:50:36 +00:00
|
|
|
"EncodeBase64": func(s string) string {
|
|
|
|
return base64.StdEncoding.EncodeToString([]byte(s))
|
|
|
|
},
|
2016-06-23 22:21:44 +00:00
|
|
|
}
|
|
|
|
|
2016-06-30 21:37:52 +00:00
|
|
|
if p.Minimal {
|
2016-12-29 22:50:36 +00:00
|
|
|
templates = append(templates, "minimal")
|
2016-06-30 21:37:52 +00:00
|
|
|
} else {
|
2016-12-29 22:50:36 +00:00
|
|
|
templates = append(templates, "base")
|
2016-06-30 21:37:52 +00:00
|
|
|
}
|
|
|
|
|
2016-06-23 21:27:13 +00:00
|
|
|
var tpl *template.Template
|
|
|
|
|
|
|
|
// For each template, add it to the the tpl variable
|
|
|
|
for i, t := range templates {
|
|
|
|
// Get the template from the assets
|
2016-10-18 20:30:10 +00:00
|
|
|
Page, err := assets.Asset("templates/" + t + ".tmpl")
|
2016-06-23 21:27:13 +00:00
|
|
|
|
|
|
|
// Check if there is some error. If so, the template doesn't exist
|
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
2016-06-10 21:18:44 +00:00
|
|
|
|
2016-06-23 21:27:13 +00:00
|
|
|
// If it's the first iteration, creates a new template and add the
|
|
|
|
// functions map
|
|
|
|
if i == 0 {
|
2016-10-18 20:30:10 +00:00
|
|
|
tpl, err = template.New(t).Funcs(functions).Parse(string(Page))
|
2016-06-23 21:27:13 +00:00
|
|
|
} else {
|
2016-10-18 20:30:10 +00:00
|
|
|
tpl, err = tpl.Parse(string(Page))
|
2016-06-23 21:27:13 +00:00
|
|
|
}
|
2016-06-21 14:02:30 +00:00
|
|
|
|
2016-06-23 21:27:13 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Print(err)
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
2016-06-21 14:02:30 +00:00
|
|
|
}
|
|
|
|
|
2016-06-22 20:21:32 +00:00
|
|
|
buf := &bytes.Buffer{}
|
2016-10-18 20:30:10 +00:00
|
|
|
err := tpl.Execute(buf, p.Info)
|
2016-06-11 09:08:33 +00:00
|
|
|
|
2016-06-11 09:37:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
2016-06-11 09:08:33 +00:00
|
|
|
}
|
|
|
|
|
2016-06-22 20:21:32 +00:00
|
|
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
|
|
|
_, err = buf.WriteTo(w)
|
2016-10-31 21:25:14 +00:00
|
|
|
return http.StatusOK, err
|
2016-06-11 09:37:36 +00:00
|
|
|
}
|
2016-06-11 09:08:33 +00:00
|
|
|
|
2016-10-18 20:30:10 +00:00
|
|
|
// PrintAsJSON prints the current Page infromation in JSON
|
|
|
|
func (p Page) PrintAsJSON(w http.ResponseWriter) (int, error) {
|
2016-11-01 15:00:36 +00:00
|
|
|
marsh, err := json.MarshalIndent(p.Info.Data, "", " ")
|
2016-06-11 09:37:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
2016-06-11 09:08:33 +00:00
|
|
|
}
|
|
|
|
|
2016-06-11 09:37:36 +00:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
2016-08-19 10:51:12 +00:00
|
|
|
if _, err := w.Write(marsh); err != nil {
|
|
|
|
return http.StatusInternalServerError, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return http.StatusOK, nil
|
2016-06-11 09:08:33 +00:00
|
|
|
}
|