A webmail client. Forked from https://git.sr.ht/~migadu/alps
Não pode escolher mais do que 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.
 
 
 
 

182 linhas
4.0 KiB

  1. package alps
  2. import (
  3. "fmt"
  4. "html/template"
  5. "io"
  6. "io/ioutil"
  7. "net/url"
  8. "os"
  9. "strings"
  10. "github.com/labstack/echo/v4"
  11. )
  12. const themesDir = "themes"
  13. // GlobalRenderData contains data available in all templates.
  14. type GlobalRenderData struct {
  15. Path []string
  16. URL *url.URL
  17. LoggedIn bool
  18. // if logged in
  19. Username string
  20. // additional plugin-specific data
  21. Extra map[string]interface{}
  22. }
  23. // BaseRenderData is the base type for templates. It should be extended with
  24. // additional template-specific fields:
  25. //
  26. // type MyRenderData struct {
  27. // BaseRenderData
  28. // // add additional fields here
  29. // }
  30. type BaseRenderData struct {
  31. GlobalData GlobalRenderData
  32. // additional plugin-specific data
  33. Extra map[string]interface{}
  34. }
  35. // Global implements RenderData.
  36. func (brd *BaseRenderData) Global() *GlobalRenderData {
  37. return &brd.GlobalData
  38. }
  39. // RenderData is implemented by template data structs. It can be used to inject
  40. // additional data to all templates.
  41. type RenderData interface {
  42. // GlobalData returns a pointer to the global render data.
  43. Global() *GlobalRenderData
  44. }
  45. // NewBaseRenderData initializes a new BaseRenderData.
  46. //
  47. // It can be used by routes to pre-fill the base data:
  48. //
  49. // type MyRenderData struct {
  50. // BaseRenderData
  51. // // add additional fields here
  52. // }
  53. //
  54. // data := &MyRenderData{
  55. // BaseRenderData: *alps.NewBaseRenderData(ctx),
  56. // // other fields...
  57. // }
  58. func NewBaseRenderData(ctx *Context) *BaseRenderData {
  59. global := GlobalRenderData{Extra: make(map[string]interface{})}
  60. if ctx.Session != nil {
  61. global.LoggedIn = true
  62. global.Username = ctx.Session.username
  63. }
  64. global.URL = ctx.Request().URL
  65. global.Path = strings.Split(global.URL.Path, "/")[1:]
  66. return &BaseRenderData{
  67. GlobalData: global,
  68. Extra: make(map[string]interface{}),
  69. }
  70. }
  71. type renderer struct {
  72. logger echo.Logger
  73. defaultTheme string
  74. base *template.Template
  75. themes map[string]*template.Template
  76. }
  77. func (r *renderer) Render(w io.Writer, name string, data interface{}, ectx echo.Context) error {
  78. // ectx is the raw *echo.context, not our own *Context
  79. ctx := ectx.Get("context").(*Context)
  80. var renderData RenderData
  81. if data == nil {
  82. renderData = &struct{ BaseRenderData }{*NewBaseRenderData(ctx)}
  83. } else {
  84. var ok bool
  85. renderData, ok = data.(RenderData)
  86. if !ok {
  87. return fmt.Errorf("data passed to template %q doesn't implement RenderData", name)
  88. }
  89. }
  90. for _, plugin := range ctx.Server.plugins {
  91. if err := plugin.Inject(ctx, name, renderData); err != nil {
  92. return fmt.Errorf("failed to run plugin %q: %v", plugin.Name(), err)
  93. }
  94. }
  95. // TODO: per-user theme selection
  96. t := r.base
  97. if r.defaultTheme != "" {
  98. t = r.themes[r.defaultTheme]
  99. }
  100. return t.ExecuteTemplate(w, name, data)
  101. }
  102. func loadTheme(name string, base *template.Template) (*template.Template, error) {
  103. theme, err := base.Clone()
  104. if err != nil {
  105. return nil, err
  106. }
  107. theme, err = theme.ParseGlob(themesDir + "/" + name + "/*.html")
  108. if err != nil {
  109. return nil, err
  110. }
  111. return theme, nil
  112. }
  113. func (r *renderer) Load(plugins []Plugin) error {
  114. base := template.New("")
  115. for _, p := range plugins {
  116. if err := p.LoadTemplate(base); err != nil {
  117. return fmt.Errorf("failed to load template for plugin %q: %v", p.Name(), err)
  118. }
  119. }
  120. themes := make(map[string]*template.Template)
  121. files, err := ioutil.ReadDir(themesDir)
  122. if err != nil && !os.IsNotExist(err) {
  123. return err
  124. }
  125. for _, fi := range files {
  126. if !fi.IsDir() {
  127. continue
  128. }
  129. r.logger.Printf("Loading theme %q", fi.Name())
  130. var err error
  131. if themes[fi.Name()], err = loadTheme(fi.Name(), base); err != nil {
  132. return fmt.Errorf("failed to load theme %q: %v", fi.Name(), err)
  133. }
  134. }
  135. if r.defaultTheme != "" {
  136. if _, ok := themes[r.defaultTheme]; !ok {
  137. return fmt.Errorf("failed to find default theme %q", r.defaultTheme)
  138. }
  139. }
  140. r.base = base
  141. r.themes = themes
  142. return nil
  143. }
  144. func newRenderer(logger echo.Logger, defaultTheme string) *renderer {
  145. return &renderer{
  146. logger: logger,
  147. defaultTheme: defaultTheme,
  148. }
  149. }