parent
dfa9bcb9e2
commit
c3135306b3
@ -0,0 +1,110 @@ |
||||
package http |
||||
|
||||
import ( |
||||
"errors" |
||||
"fmt" |
||||
"net/http" |
||||
"path/filepath" |
||||
"strings" |
||||
|
||||
"github.com/mediocregopher/blog.mediocregopher.com/srv/http/apiutil" |
||||
"github.com/mediocregopher/blog.mediocregopher.com/srv/post" |
||||
) |
||||
|
||||
func (a *api) renderDraftPostHandler() http.Handler { |
||||
|
||||
tpl := a.mustParseBasedTpl("post.html") |
||||
renderDraftPostsIndexHandler := a.renderDraftPostsIndexHandler() |
||||
renderDraftEditPostHandler := a.renderEditPostHandler(true) |
||||
|
||||
return http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) { |
||||
|
||||
id := strings.TrimSuffix(filepath.Base(r.URL.Path), ".html") |
||||
|
||||
if id == "/" { |
||||
renderDraftPostsIndexHandler.ServeHTTP(rw, r) |
||||
return |
||||
} |
||||
|
||||
if _, ok := r.URL.Query()["edit"]; ok { |
||||
renderDraftEditPostHandler.ServeHTTP(rw, r) |
||||
return |
||||
} |
||||
|
||||
p, err := a.params.PostDraftStore.GetByID(id) |
||||
|
||||
if errors.Is(err, post.ErrPostNotFound) { |
||||
http.Error(rw, "Post not found", 404) |
||||
return |
||||
} else if err != nil { |
||||
apiutil.InternalServerError( |
||||
rw, r, fmt.Errorf("fetching post with id %q: %w", id, err), |
||||
) |
||||
return |
||||
} |
||||
|
||||
tplPayload, err := a.postToPostTplPayload(post.StoredPost{Post: p}) |
||||
|
||||
if err != nil { |
||||
apiutil.InternalServerError( |
||||
rw, r, fmt.Errorf( |
||||
"generating template payload for post with id %q: %w", |
||||
id, err, |
||||
), |
||||
) |
||||
return |
||||
} |
||||
|
||||
executeTemplate(rw, r, tpl, tplPayload) |
||||
}) |
||||
} |
||||
|
||||
func (a *api) renderDraftPostsIndexHandler() http.Handler { |
||||
|
||||
renderEditPostHandler := a.renderEditPostHandler(true) |
||||
tpl := a.mustParseBasedTpl("draft-posts.html") |
||||
const pageCount = 20 |
||||
|
||||
return http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) { |
||||
|
||||
if _, ok := r.URL.Query()["edit"]; ok { |
||||
renderEditPostHandler.ServeHTTP(rw, r) |
||||
return |
||||
} |
||||
|
||||
page, err := apiutil.StrToInt(r.FormValue("p"), 0) |
||||
if err != nil { |
||||
apiutil.BadRequest( |
||||
rw, r, fmt.Errorf("invalid page number: %w", err), |
||||
) |
||||
return |
||||
} |
||||
|
||||
posts, hasMore, err := a.params.PostDraftStore.Get(page, pageCount) |
||||
if err != nil { |
||||
apiutil.InternalServerError( |
||||
rw, r, fmt.Errorf("fetching page %d of posts: %w", page, err), |
||||
) |
||||
return |
||||
} |
||||
|
||||
tplPayload := struct { |
||||
Posts []post.Post |
||||
PrevPage, NextPage int |
||||
}{ |
||||
Posts: posts, |
||||
PrevPage: -1, |
||||
NextPage: -1, |
||||
} |
||||
|
||||
if page > 0 { |
||||
tplPayload.PrevPage = page - 1 |
||||
} |
||||
|
||||
if hasMore { |
||||
tplPayload.NextPage = page + 1 |
||||
} |
||||
|
||||
executeTemplate(rw, r, tpl, tplPayload) |
||||
}) |
||||
} |
@ -0,0 +1,48 @@ |
||||
{{ define "body" }} |
||||
|
||||
<h1>Drafts</h1> |
||||
|
||||
<p> |
||||
<a href="{{ BlogURL "drafts/" }}?edit"> |
||||
New Draft |
||||
</a> |
||||
</p> |
||||
|
||||
{{ if ge .Payload.PrevPage 0 }} |
||||
<p> |
||||
<a href="?p={{ .Payload.PrevPage}}">< < Previous Page</a> |
||||
</p> |
||||
{{ end }} |
||||
|
||||
<table> |
||||
|
||||
{{ range .Payload.Posts }} |
||||
<tr> |
||||
<td><a href="{{ DraftURL .ID }}">{{ .Title }}</a></td> |
||||
<td> |
||||
<a href="{{ DraftURL .ID }}?edit"> |
||||
Edit |
||||
</a> |
||||
</td> |
||||
<td> |
||||
<form |
||||
action="{{ DraftURL .ID }}?method=delete" |
||||
method="POST" |
||||
> |
||||
<input type="submit" value="Delete" /> |
||||
</form> |
||||
</td> |
||||
</tr> |
||||
{{ end }} |
||||
|
||||
</table> |
||||
|
||||
{{ if ge .Payload.NextPage 0 }} |
||||
<p> |
||||
<a href="?p={{ .Payload.NextPage}}">Next Page > ></a> |
||||
</p> |
||||
{{ end }} |
||||
|
||||
{{ end }} |
||||
|
||||
{{ template "base.html" . }} |
Loading…
Reference in new issue