A webmail client. Forked from https://git.sr.ht/~migadu/alps
選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 

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