Generalize sidebar categorization logic

This commit is contained in:
Drew DeVault 2020-10-22 13:03:50 -04:00
parent 9e2c420461
commit 6ba418c9b2
5 changed files with 149 additions and 143 deletions

View file

@ -63,12 +63,16 @@ func registerRoutes(p *alps.GoPlugin) {
p.POST("/settings", handleSettings) p.POST("/settings", handleSettings)
} }
type MailboxRenderData struct { type IMAPBaseRenderData struct {
alps.BaseRenderData alps.BaseRenderData
Mailbox *MailboxStatus
Inbox *MailboxStatus
CategorizedMailboxes CategorizedMailboxes CategorizedMailboxes CategorizedMailboxes
Mailboxes []MailboxInfo Mailboxes []MailboxInfo
Mailbox *MailboxStatus
Inbox *MailboxStatus
}
type MailboxRenderData struct {
IMAPBaseRenderData
Messages []IMAPMessage Messages []IMAPMessage
PrevPage, NextPage int PrevPage, NextPage int
Query string Query string
@ -87,21 +91,52 @@ type CategorizedMailboxes struct {
Additional []*MailboxInfo Additional []*MailboxInfo
} }
func categorizeMailboxes(mailboxes []MailboxInfo, func newIMAPBaseRenderData(ctx *alps.Context,
inbox *MailboxStatus, active *MailboxStatus) CategorizedMailboxes { base *alps.BaseRenderData) (*IMAPBaseRenderData, error) {
var out CategorizedMailboxes mboxName, err := url.PathUnescape(ctx.Param("mbox"))
mmap := map[string]**MailboxInfo{ if err != nil {
"INBOX": &out.Common.Inbox, return nil, echo.NewHTTPError(http.StatusBadRequest, err)
"Drafts": &out.Common.Drafts,
"Sent": &out.Common.Sent,
"Junk": &out.Common.Junk,
"Trash": &out.Common.Trash,
"Archive": &out.Common.Archive,
} }
var mailboxes []MailboxInfo
var active, inbox *MailboxStatus
err = ctx.Session.DoIMAP(func(c *imapclient.Client) error {
var err error
if mailboxes, err = listMailboxes(c); err != nil {
return err
}
if mboxName != "" {
if active, err = getMailboxStatus(c, mboxName); err != nil {
return err
}
}
if mboxName == "INBOX" {
inbox = active
} else {
if inbox, err = getMailboxStatus(c, "INBOX"); err != nil {
return err
}
}
return nil
})
if err != nil {
return nil, err
}
var categorized CategorizedMailboxes
mmap := map[string]**MailboxInfo{
"INBOX": &categorized.Common.Inbox,
"Drafts": &categorized.Common.Drafts,
"Sent": &categorized.Common.Sent,
"Junk": &categorized.Common.Junk,
"Trash": &categorized.Common.Trash,
"Archive": &categorized.Common.Archive,
}
for i, _ := range mailboxes { for i, _ := range mailboxes {
// Populate unseen & active states // Populate unseen & active states
if mailboxes[i].Name == active.Name { if active != nil && mailboxes[i].Name == active.Name {
mailboxes[i].Unseen = int(active.Unseen) mailboxes[i].Unseen = int(active.Unseen)
mailboxes[i].Active = true mailboxes[i].Active = true
} }
@ -112,18 +147,36 @@ func categorizeMailboxes(mailboxes []MailboxInfo,
if ptr, ok := mmap[mailboxes[i].Name]; ok { if ptr, ok := mmap[mailboxes[i].Name]; ok {
*ptr = &mailboxes[i] *ptr = &mailboxes[i]
} else { } else {
out.Additional = append(out.Additional, &mailboxes[i]) categorized.Additional = append(
categorized.Additional, &mailboxes[i])
} }
} }
return out
return &IMAPBaseRenderData{
BaseRenderData: *base,
CategorizedMailboxes: categorized,
Mailboxes: mailboxes,
Inbox: inbox,
Mailbox: active,
}, nil
} }
func handleGetMailbox(ctx *alps.Context) error { func handleGetMailbox(ctx *alps.Context) error {
mboxName, err := url.PathUnescape(ctx.Param("mbox")) ibase, err := newIMAPBaseRenderData(ctx, alps.NewBaseRenderData(ctx))
if err != nil { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, err) return err
} }
mbox := ibase.Mailbox
title := mbox.Name
if title == "INBOX" {
title = "Inbox"
}
if mbox.Unseen > 0 {
title = fmt.Sprintf("(%d) %s", mbox.Unseen, title)
}
ibase.BaseRenderData.WithTitle(title)
page := 0 page := 0
if pageStr := ctx.QueryParam("page"); pageStr != "" { if pageStr := ctx.QueryParam("page"); pageStr != "" {
var err error var err error
@ -140,33 +193,20 @@ func handleGetMailbox(ctx *alps.Context) error {
query := ctx.QueryParam("query") query := ctx.QueryParam("query")
var mailboxes []MailboxInfo var (
var msgs []IMAPMessage msgs []IMAPMessage
var mbox, inbox *MailboxStatus total int
var total int )
err = ctx.Session.DoIMAP(func(c *imapclient.Client) error { err = ctx.Session.DoIMAP(func(c *imapclient.Client) error {
var err error var err error
if mailboxes, err = listMailboxes(c); err != nil {
return err
}
if mbox, err = getMailboxStatus(c, mboxName); err != nil {
return err
}
if query != "" { if query != "" {
msgs, total, err = searchMessages(c, mboxName, query, page, messagesPerPage) msgs, total, err = searchMessages(c, mbox.Name, query, page, messagesPerPage)
} else { } else {
msgs, err = listMessages(c, mbox, page, messagesPerPage) msgs, err = listMessages(c, mbox, page, messagesPerPage)
} }
if err != nil { if err != nil {
return err return err
} }
if mboxName == "INBOX" {
inbox = mbox
} else {
if inbox, err = getMailboxStatus(c, "INBOX"); err != nil {
return err
}
}
return nil return nil
}) })
if err != nil { if err != nil {
@ -190,23 +230,8 @@ func handleGetMailbox(ctx *alps.Context) error {
} }
} }
title := mbox.Name
if title == "INBOX" {
title = "Inbox"
}
if mbox.Unseen > 0 {
title = fmt.Sprintf("(%d) %s", mbox.Unseen, title)
}
categorized := categorizeMailboxes(mailboxes, inbox, mbox)
return ctx.Render(http.StatusOK, "mailbox.html", &MailboxRenderData{ return ctx.Render(http.StatusOK, "mailbox.html", &MailboxRenderData{
BaseRenderData: *alps.NewBaseRenderData(ctx).WithTitle(title), IMAPBaseRenderData: *ibase,
Mailbox: mbox,
Inbox: inbox,
CategorizedMailboxes: categorized,
Mailboxes: mailboxes,
Messages: msgs, Messages: msgs,
PrevPage: prevPage, PrevPage: prevPage,
NextPage: nextPage, NextPage: nextPage,
@ -262,10 +287,7 @@ func handleLogout(ctx *alps.Context) error {
} }
type MessageRenderData struct { type MessageRenderData struct {
alps.BaseRenderData IMAPBaseRenderData
Mailboxes []MailboxInfo
Mailbox *MailboxStatus
Inbox *MailboxStatus
Message *IMAPMessage Message *IMAPMessage
Part *IMAPPartNode Part *IMAPPartNode
View interface{} View interface{}
@ -274,10 +296,13 @@ type MessageRenderData struct {
} }
func handleGetPart(ctx *alps.Context, raw bool) error { func handleGetPart(ctx *alps.Context, raw bool) error {
mboxName, uid, err := parseMboxAndUid(ctx.Param("mbox"), ctx.Param("uid")) _, uid, err := parseMboxAndUid(ctx.Param("mbox"), ctx.Param("uid"))
ibase, err := newIMAPBaseRenderData(ctx, alps.NewBaseRenderData(ctx))
if err != nil { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, err) return err
} }
mbox := ibase.Mailbox
partPath, err := parsePartPath(ctx.QueryParam("part")) partPath, err := parsePartPath(ctx.QueryParam("part"))
if err != nil { if err != nil {
return echo.NewHTTPError(http.StatusBadRequest, err) return echo.NewHTTPError(http.StatusBadRequest, err)
@ -289,28 +314,13 @@ func handleGetPart(ctx *alps.Context, raw bool) error {
} }
messagesPerPage := settings.MessagesPerPage messagesPerPage := settings.MessagesPerPage
var mailboxes []MailboxInfo
var msg *IMAPMessage var msg *IMAPMessage
var part *message.Entity var part *message.Entity
var mbox, inbox *MailboxStatus
err = ctx.Session.DoIMAP(func(c *imapclient.Client) error { err = ctx.Session.DoIMAP(func(c *imapclient.Client) error {
var err error var err error
if mailboxes, err = listMailboxes(c); err != nil { if msg, part, err = getMessagePart(c, mbox.Name, uid, partPath); err != nil {
return err return err
} }
if msg, part, err = getMessagePart(c, mboxName, uid, partPath); err != nil {
return err
}
if mbox, err = getMailboxStatus(c, mboxName); err != nil {
return err
}
if mboxName == "INBOX" {
inbox = mbox
} else {
if inbox, err = getMailboxStatus(c, "INBOX"); err != nil {
return err
}
}
return nil return nil
}) })
if err != nil { if err != nil {
@ -367,12 +377,10 @@ func handleGetPart(ctx *alps.Context, raw bool) error {
flags[f] = msg.HasFlag(f) flags[f] = msg.HasFlag(f)
} }
ibase.BaseRenderData.WithTitle(msg.Envelope.Subject)
return ctx.Render(http.StatusOK, "message.html", &MessageRenderData{ return ctx.Render(http.StatusOK, "message.html", &MessageRenderData{
BaseRenderData: *alps.NewBaseRenderData(ctx). IMAPBaseRenderData: *ibase,
WithTitle(msg.Envelope.Subject),
Mailboxes: mailboxes,
Mailbox: mbox,
Inbox: inbox,
Message: msg, Message: msg,
Part: msg.PartByPath(partPath), Part: msg.PartByPath(partPath),
View: view, View: view,
@ -382,7 +390,7 @@ func handleGetPart(ctx *alps.Context, raw bool) error {
} }
type ComposeRenderData struct { type ComposeRenderData struct {
alps.BaseRenderData IMAPBaseRenderData
Message *OutgoingMessage Message *OutgoingMessage
} }
@ -438,6 +446,11 @@ func submitCompose(ctx *alps.Context, msg *OutgoingMessage, options *composeOpti
} }
func handleCompose(ctx *alps.Context, msg *OutgoingMessage, options *composeOptions) error { func handleCompose(ctx *alps.Context, msg *OutgoingMessage, options *composeOptions) error {
ibase, err := newIMAPBaseRenderData(ctx, alps.NewBaseRenderData(ctx))
if err != nil {
return err
}
if msg.From == "" && strings.ContainsRune(ctx.Session.Username(), '@') { if msg.From == "" && strings.ContainsRune(ctx.Session.Username(), '@') {
msg.From = ctx.Session.Username() msg.From = ctx.Session.Username()
} }
@ -537,7 +550,7 @@ func handleCompose(ctx *alps.Context, msg *OutgoingMessage, options *composeOpti
} }
return ctx.Render(http.StatusOK, "compose.html", &ComposeRenderData{ return ctx.Render(http.StatusOK, "compose.html", &ComposeRenderData{
BaseRenderData: *alps.NewBaseRenderData(ctx), IMAPBaseRenderData: *ibase,
Message: msg, Message: msg,
}) })
} }

View file

@ -1,18 +1,9 @@
{{template "head.html" .}} {{template "head.html" .}}
{{template "nav.html" .}} {{template "nav.html" .}}
{{template "util.html" .}}
<div class="page-wrap"> <div class="page-wrap">
<aside> {{ template "aside" . }}
<a href="/compose" class="new active">Compose&nbsp;Mail</a>
<!-- TODO: use mailbox list from template data -->
<a href="/mailbox/INBOX">Inbox</a>
<a href="/mailbox/Drafts">Drafts</a>
<a href="/mailbox/Sent">Sent</a>
<a href="/mailbox/Archive">Archive</a>
<a href="/mailbox/Junk">Junk</a>
<a href="/mailbox/Trash">Trash</a>
</aside>
<div class="container"> <div class="container">
<main class="create-update"> <main class="create-update">

View file

@ -1,5 +1,6 @@
{{template "head.html" .}} {{template "head.html" .}}
{{template "nav.html" .}} {{template "nav.html" .}}
{{template "util.html" .}}
{{ define "mbox-link" }} {{ define "mbox-link" }}
{{ if not (.HasAttr "\\Noselect") }} {{ if not (.HasAttr "\\Noselect") }}
@ -21,25 +22,7 @@
{{ end }} {{ end }}
<div class="page-wrap"> <div class="page-wrap">
<aside> {{ template "aside" . }}
<!-- the logo image, dimensions 200x32 may be present or not -->
<a href="/compose" class="new">Compose&nbsp;Mail</a>
{{ with .CategorizedMailboxes }}
{{ with .Common.Inbox }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Drafts }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Sent }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Junk }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Trash }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Archive }}{{ template "mbox-link" . }}{{ end}}
{{ if .Additional }}
<hr />
{{ range .Additional }}
{{ template "mbox-link" . }}
{{ end }}
{{ end }}
{{ end }}
</aside>
<div class="container"> <div class="container">
<form id="messages-form" method="post"></form> <form id="messages-form" method="post"></form>
<main class="message-list"> <main class="message-list">

View file

@ -1,5 +1,6 @@
{{template "head.html" .}} {{template "head.html" .}}
{{template "nav.html" .}} {{template "nav.html" .}}
{{template "util.html" .}}
{{define "message-part-tree"}} {{define "message-part-tree"}}
{{/* nested templates can't access the parent's context */}} {{/* nested templates can't access the parent's context */}}
@ -28,31 +29,8 @@
{{end}} {{end}}
<div class="page-wrap"> <div class="page-wrap">
{{$current := .Mailbox}} {{ $current := .Mailbox }}
<aside> {{ template "aside" . }}
<!-- the logo image, dimensions 200x32 may be present or not -->
<a href="/compose" class="new">Compose&nbsp;Mail</a>
{{range .Mailboxes}}
<a href="{{.URL}}"
{{ if eq $current.Name .Name }}class="active"{{ end }}>
{{ if eq .Name "INBOX" }}
Inbox
{{else}}
{{.Name}}
{{end}}
{{ $unseen := 0 }}
{{ if eq .Name "INBOX" }}
{{ $unseen = $.Inbox.Unseen }}
{{ end }}
{{ if eq .Name $.Mailbox.Name }}
{{ $unseen = $.Mailbox.Unseen }}
{{ end }}
{{ if $unseen }}({{ $unseen }}){{ end }}
</a>
{{end}}
</aside>
<div class="container"> <div class="container">
<main class="message"> <main class="message">
<section class="actions"> <section class="actions">

41
themes/alps/util.html Normal file
View file

@ -0,0 +1,41 @@
{{ define "mbox-link" }}
{{ if not (.HasAttr "\\Noselect") }}
<a href="{{.URL}}" {{ if .Active }}class="active"{{ end }}>
{{- if eq .Name "INBOX" -}}
Inbox
{{- else -}}
{{ .Name }}
{{- end -}}
{{- if .HasAttr "\\HasChildren" }}/{{ end }}
{{ if and (ne .Unseen -1) (ne .Unseen 0) }}({{ .Unseen }}){{ end }}
</a>
{{ else }}
<span class="noselect">
{{.Name}}{{- if .HasAttr "\\HasChildren" }}/{{ end }}
</span>
{{ end }}
{{ end }}
{{ define "aside" }}
<aside>
<!-- the logo image, dimensions 200x32 may be present or not -->
<a href="/compose" class="new
{{ if eq $.GlobalData.URL.Path "/compose" }}active{{ end }}
">Compose&nbsp;Mail</a>
{{ with .CategorizedMailboxes }}
{{ with .Common.Inbox }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Drafts }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Sent }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Junk }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Trash }}{{ template "mbox-link" . }}{{ end}}
{{ with .Common.Archive }}{{ template "mbox-link" . }}{{ end}}
{{ if .Additional }}
<hr />
{{ range .Additional }}
{{ template "mbox-link" . }}
{{ end }}
{{ end }}
{{ end }}
</aside>
{{ end }}