generated from ElnuDev/go-project
Add hot tempalte reloading, WIP pos
This commit is contained in:
parent
e00df6aac6
commit
442f727d8a
4 changed files with 73 additions and 15 deletions
20
dict/main.go
20
dict/main.go
|
@ -30,7 +30,12 @@ func LoadDict() error {
|
||||||
type Entry struct {
|
type Entry struct {
|
||||||
Kanji string
|
Kanji string
|
||||||
Reading string
|
Reading string
|
||||||
Definitions []string
|
Definitions []Definition
|
||||||
|
}
|
||||||
|
|
||||||
|
type Definition struct {
|
||||||
|
Definition string
|
||||||
|
PartOfSpeech string
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseEntry(entry jmdict.JmdictEntry) Entry {
|
func ParseEntry(entry jmdict.JmdictEntry) Entry {
|
||||||
|
@ -42,11 +47,18 @@ func ParseEntry(entry jmdict.JmdictEntry) Entry {
|
||||||
if len(entry.Readings) > 0 {
|
if len(entry.Readings) > 0 {
|
||||||
reading = entry.Readings[0].Reading
|
reading = entry.Readings[0].Reading
|
||||||
}
|
}
|
||||||
var definitions []string
|
var definitions []Definition
|
||||||
if len(entry.Sense) > 0 && len(entry.Sense[0].Glossary) > 0 {
|
if len(entry.Sense) > 0 && len(entry.Sense[0].Glossary) > 0 {
|
||||||
definitions = make([]string, len(entry.Sense[0].Glossary))
|
definitions = make([]Definition, len(entry.Sense[0].Glossary))
|
||||||
for i, glossary := range entry.Sense[0].Glossary {
|
for i, glossary := range entry.Sense[0].Glossary {
|
||||||
definitions[i] = glossary.Content
|
var pos string
|
||||||
|
if len(entry.Sense[0].PartsOfSpeech) > i+1 {
|
||||||
|
pos = entry.Sense[0].PartsOfSpeech[i]
|
||||||
|
}
|
||||||
|
definitions[i] = Definition{
|
||||||
|
Definition: glossary.Content,
|
||||||
|
PartOfSpeech: pos,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Entry{
|
return Entry{
|
||||||
|
|
3
dict/templates/definition.html
Normal file
3
dict/templates/definition.html
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
{{ define "definition" }}
|
||||||
|
{{ if .PartOfSpeech }}<chip>{{ .PartOfSpeech }}</chip> {{ end }}{{ .Definition -}}
|
||||||
|
{{ end }}
|
|
@ -9,11 +9,11 @@
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
</h3>
|
</h3>
|
||||||
{{ if le (len .Definitions) 2 -}}
|
{{ if le (len .Definitions) 2 -}}
|
||||||
<p>{{- index .Definitions 0 -}}</p>
|
<p>{{- template "definition" (index .Definitions 0) -}}</p>
|
||||||
{{- else -}}
|
{{- else -}}
|
||||||
<ol>
|
<ol>
|
||||||
{{- range .Definitions }}
|
{{- range .Definitions }}
|
||||||
<li>{{- . -}}</li>
|
<li>{{- template "definition" . -}}</li>
|
||||||
{{- end }}
|
{{- end }}
|
||||||
</ol>
|
</ol>
|
||||||
{{- end }}
|
{{- end }}
|
||||||
|
|
|
@ -3,25 +3,68 @@ package httputils
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
"text/template"
|
"text/template"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Handler = func(http.ResponseWriter, *http.Request)
|
type Handler = func(http.ResponseWriter, *http.Request)
|
||||||
|
|
||||||
|
const templateFolder = "templates"
|
||||||
|
|
||||||
|
var templatePaths, templateModTimes, _ = getTemplates()
|
||||||
|
var templates *template.Template = template.Must(template.ParseFiles(templatePaths...))
|
||||||
|
|
||||||
|
func getTemplates() ([]string, map[string]time.Time, error) {
|
||||||
|
var modTimes map[string]time.Time = make(map[string]time.Time)
|
||||||
|
err := filepath.Walk(templateFolder, func(path string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !info.IsDir() && strings.HasSuffix(path, ".html") {
|
||||||
|
modTimes[path] = info.ModTime()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
paths := make([]string, len(modTimes))
|
||||||
|
i := 0
|
||||||
|
for path := range modTimes {
|
||||||
|
paths[i] = path
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
return paths, modTimes, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func reloadTemplateIfModified(path string) {
|
||||||
|
fileInfo, _ := os.Stat(path)
|
||||||
|
modTime := fileInfo.ModTime()
|
||||||
|
if modTime.After(templateModTimes[path]) {
|
||||||
|
fmt.Printf("Reloading template %s...\n", path)
|
||||||
|
templates.ParseFiles(path)
|
||||||
|
templateModTimes[path] = modTime
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func reloadTemplatesIfModified() {
|
||||||
|
for _, path := range templatePaths {
|
||||||
|
reloadTemplateIfModified(path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const reloadTemplates = true
|
||||||
|
|
||||||
func GenerateHandler(
|
func GenerateHandler(
|
||||||
file string,
|
file string,
|
||||||
handler func(http.ResponseWriter, *http.Request) bool,
|
handler func(http.ResponseWriter, *http.Request) bool,
|
||||||
data func(http.ResponseWriter, *http.Request) any,
|
data func(http.ResponseWriter, *http.Request) any,
|
||||||
methods []string,
|
methods []string,
|
||||||
) Handler {
|
) Handler {
|
||||||
var tmpl *template.Template
|
|
||||||
//if file != "" {
|
|
||||||
// tmpl = template.Must(template.ParseFiles(fmt.Sprintf("templates/%s", file)))
|
|
||||||
//}
|
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
// Remove in production, enables live template reloading
|
// All templates must be reloaded in case of dependencies
|
||||||
if file != "" {
|
if reloadTemplates {
|
||||||
tmpl = template.Must(template.ParseFiles(fmt.Sprintf("templates/%s", file)))
|
reloadTemplatesIfModified()
|
||||||
}
|
}
|
||||||
for _, method := range methods {
|
for _, method := range methods {
|
||||||
if method == r.Method {
|
if method == r.Method {
|
||||||
|
@ -32,9 +75,9 @@ func GenerateHandler(
|
||||||
return
|
return
|
||||||
ok:
|
ok:
|
||||||
renderTemplate := handler(w, r)
|
renderTemplate := handler(w, r)
|
||||||
if renderTemplate && tmpl != nil {
|
if renderTemplate && file != "" {
|
||||||
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
||||||
tmpl.Execute(w, data(w, r))
|
templates.ExecuteTemplate(w, file, data(w, r))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Reference in a new issue