summaryrefslogtreecommitdiff
path: root/view
diff options
context:
space:
mode:
Diffstat (limited to 'view')
-rw-r--r--view/common.go64
-rw-r--r--view/html/footer.html5
-rw-r--r--view/html/head.html8
-rw-r--r--view/html/index.html40
4 files changed, 117 insertions, 0 deletions
diff --git a/view/common.go b/view/common.go
new file mode 100644
index 0000000..c371fa0
--- /dev/null
+++ b/view/common.go
@@ -0,0 +1,64 @@
+package view
+
+import (
+ "embed"
+ "html/template"
+ "net/http"
+ "strings"
+ "reflect"
+
+ "xengineering.eu/ceres/model"
+
+ "github.com/gorilla/mux"
+)
+
+//go:embed html/*.html
+var htmlFS embed.FS
+
+var html *template.Template
+
+func Init() {
+ html = template.Must(template.New("html").ParseFS(htmlFS, "html/*.html"))
+}
+
+func HandlerHTML(prototype model.ReadableData) http.HandlerFunc {
+ t := reflect.TypeOf(prototype).Elem()
+
+ tmpl := t.String()
+ tmpl = strings.TrimPrefix(tmpl, `model.`)
+ tmpl = strings.ToLower(tmpl)
+
+ return func(w http.ResponseWriter, r *http.Request) {
+ data := reflect.New(t).Interface().(model.ReadableData)
+ var err error
+
+ v := reflect.ValueOf(data).Elem()
+ if v.Kind() == reflect.Struct {
+ id, ok := mux.Vars(r)[`id`]
+ if ok {
+ f := v.FieldByName(`Id`)
+ if f.IsValid() && f.CanSet() && f.Kind() == reflect.String {
+ f.SetString(id)
+ } else {
+ http.Error(w, `Requested struct data does not have a settable string ID`, http.StatusBadRequest)
+ return
+ }
+ } else {
+ http.Error(w, `Requested struct data without giving an ID`, http.StatusBadRequest)
+ return
+ }
+ }
+
+ err = data.FromDB()
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+
+ err = html.ExecuteTemplate(w, tmpl, data)
+ if err != nil {
+ http.Error(w, err.Error(), http.StatusInternalServerError)
+ return
+ }
+ }
+}
diff --git a/view/html/footer.html b/view/html/footer.html
new file mode 100644
index 0000000..5128b7f
--- /dev/null
+++ b/view/html/footer.html
@@ -0,0 +1,5 @@
+{{define "footer"}}
+<footer>
+ <p>The <a href="https://xengineering.eu/git/ceres">Ceres</a> recipe server is licensed under <a href="https://www.gnu.org/licenses/agpl-3.0.en.html">AGPL v3</a> and developed with <a href="https://simplecss.org/">simple.css</a>.</p>
+</footer>
+{{end}}
diff --git a/view/html/head.html b/view/html/head.html
new file mode 100644
index 0000000..91ae004
--- /dev/null
+++ b/view/html/head.html
@@ -0,0 +1,8 @@
+{{define "head"}}
+<head>
+ <title>Recipes</title>
+ <meta charset="utf-8"/>
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
+ <link rel="stylesheet" href="/static/view/static/simple.css/simple.css" type="text/css">
+</head>
+{{end}}
diff --git a/view/html/index.html b/view/html/index.html
new file mode 100644
index 0000000..45f600f
--- /dev/null
+++ b/view/html/index.html
@@ -0,0 +1,40 @@
+{{define "index"}}
+<html>
+ {{ template "head" }}
+ <header>
+ <nav>
+ <a href="/">HOME</a>
+ </nav>
+ <h1>Recipe overview</h1>
+ </header>
+ <body>
+ <main>
+ <p>Here are the available recipes 😋🍳🍔🍕🥘</p>
+ <input id="search" onkeyup="filter()" type="text" placeholder="Search for a recipe ..."></input>
+ <ul id="recipes">{{range .}}
+ <li><a href="./recipes/{{.Id}}">{{.Title}}</a></li>{{end}}
+ </ul>
+ </main>
+ {{ template "footer" }}
+ <script>
+ function filter() {
+ var input, query, ul, li, a, i, txtValue;
+ input = document.getElementById('search');
+ query = input.value.toUpperCase();
+ ul = document.getElementById("recipes");
+ li = ul.getElementsByTagName('li');
+
+ for (i = 0; i < li.length; i++) {
+ a = li[i].getElementsByTagName("a")[0];
+ txtValue = a.textContent || a.innerText;
+ if (txtValue.toUpperCase().indexOf(query) > -1) {
+ li[i].style.display = "";
+ } else {
+ li[i].style.display = "none";
+ }
+ }
+ }
+ </script>
+ </body>
+</html>
+{{end}}