2018-02-26 09:07:00 +00:00
|
|
|
package routes
|
|
|
|
|
|
|
|
import (
|
2022-02-21 03:32:53 +00:00
|
|
|
"database/sql"
|
|
|
|
"math"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
c "github.com/Azareal/Gosora/common"
|
|
|
|
"github.com/Azareal/Gosora/common/phrases"
|
|
|
|
qgen "github.com/Azareal/Gosora/query_gen"
|
2018-02-26 09:07:00 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type ProfileStmts struct {
|
2022-02-21 03:32:53 +00:00
|
|
|
getReplies *sql.Stmt
|
2018-02-26 09:07:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var profileStmts ProfileStmts
|
|
|
|
|
|
|
|
// TODO: Move these DbInits into some sort of abstraction
|
|
|
|
func init() {
|
2022-02-21 03:32:53 +00:00
|
|
|
c.DbInits.Add(func(acc *qgen.Accumulator) error {
|
|
|
|
profileStmts = ProfileStmts{
|
|
|
|
getReplies: acc.SimpleLeftJoin("users_replies", "users", "users_replies.rid, users_replies.content, users_replies.createdBy, users_replies.createdAt, users_replies.lastEdit, users_replies.lastEditBy, users.avatar, users.name, users.group", "users_replies.createdBy=users.uid", "users_replies.uid=?", "", ""),
|
|
|
|
}
|
|
|
|
return acc.FirstError()
|
|
|
|
})
|
2018-02-26 09:07:00 +00:00
|
|
|
}
|
|
|
|
|
2018-10-14 05:08:44 +00:00
|
|
|
// TODO: Remove the View part of the name?
|
2020-04-27 12:41:55 +00:00
|
|
|
func ViewProfile(w http.ResponseWriter, r *http.Request, user *c.User, h *c.Header) c.RouteError {
|
2022-02-21 03:32:53 +00:00
|
|
|
var reCreatedAt time.Time
|
|
|
|
var reContent, reCreatedByName, reAvatar string
|
|
|
|
var rid, reCreatedBy, reLastEdit, reLastEditBy, reGroup int
|
|
|
|
var reList []*c.ReplyUser
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
// TODO: Do a 301 if it's the wrong username? Do a canonical too?
|
|
|
|
_, pid, err := ParseSEOURL(r.URL.Path[len("/user/"):])
|
|
|
|
if err != nil {
|
|
|
|
return c.SimpleError(phrases.GetErrorPhrase("url_id_must_be_integer"), w, r, h)
|
|
|
|
}
|
|
|
|
if pid == 0 {
|
|
|
|
return c.NotFound(w, r, h)
|
|
|
|
}
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
var puser *c.User
|
|
|
|
if pid == user.ID {
|
|
|
|
user.IsMod = true
|
|
|
|
puser = user
|
|
|
|
} else {
|
|
|
|
// Fetch the user data
|
|
|
|
// TODO: Add a shared function for checking for ErrNoRows and internal erroring if it's not that case?
|
|
|
|
puser, err = c.Users.Get(pid)
|
|
|
|
if err == sql.ErrNoRows {
|
|
|
|
return c.NotFound(w, r, h)
|
|
|
|
} else if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
h.Title = phrases.GetTitlePhrasef("profile", puser.Name)
|
|
|
|
h.Path = c.BuildProfileURL(c.NameToSlug(puser.Name), puser.ID)
|
|
|
|
// TODO: Preload this?
|
|
|
|
h.AddSheet(h.Theme.Name + "/profile.css")
|
|
|
|
if user.Loggedin {
|
|
|
|
h.AddScriptAsync("profile_member.js")
|
|
|
|
}
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
// Get the replies..
|
|
|
|
rows, err := profileStmts.getReplies.Query(puser.ID)
|
|
|
|
if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
for rows.Next() {
|
|
|
|
err := rows.Scan(&rid, &reContent, &reCreatedBy, &reCreatedAt, &reLastEdit, &reLastEditBy, &reAvatar, &reCreatedByName, &reGroup)
|
|
|
|
if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
reLiked := false
|
|
|
|
reLikeCount := 0
|
|
|
|
ru := &c.ReplyUser{Reply: c.Reply{rid, puser.ID, reContent, reCreatedBy /*, reGroup*/, reCreatedAt, reLastEdit, reLastEditBy, 0, "", reLiked, reLikeCount, 0, ""}, ContentHtml: c.ParseMessage(reContent, 0, "", user.ParseSettings, user), CreatedByName: reCreatedByName, Avatar: reAvatar, Group: reGroup, Level: 0}
|
|
|
|
_, err = ru.Init(user)
|
|
|
|
if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
|
|
|
if puser.ID == ru.CreatedBy {
|
|
|
|
ru.Tag = phrases.GetTmplPhrase("profile.owner_tag")
|
|
|
|
}
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
// TODO: Add a hook here
|
|
|
|
reList = append(reList, ru)
|
|
|
|
}
|
|
|
|
if err := rows.Err(); err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
2018-02-26 09:07:00 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
// Normalise the score so that the user sees their relative progress to the next level rather than showing them their total score
|
|
|
|
prevScore := c.GetLevelScore(puser.Level)
|
|
|
|
score := puser.Score
|
|
|
|
//score = 23
|
|
|
|
currentScore := score - prevScore
|
|
|
|
nextScore := c.GetLevelScore(puser.Level+1) - prevScore
|
|
|
|
perc := int(math.Floor((float64(currentScore) / float64(nextScore)) * 100))
|
|
|
|
var blocked, blockedInv bool
|
|
|
|
if user.Loggedin {
|
|
|
|
blocked, err = c.UserBlocks.IsBlockedBy(user.ID, puser.ID)
|
|
|
|
if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
|
|
|
blockedInv, err = c.UserBlocks.IsBlockedBy(puser.ID, user.ID)
|
|
|
|
if err != nil {
|
|
|
|
return c.InternalError(err, w, r)
|
|
|
|
}
|
|
|
|
}
|
2020-07-14 21:50:29 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
canMessage := (!blockedInv && user.Perms.UseConvos) || (!blockedInv && puser.IsSuperMod && user.Perms.UseConvosOnlyWithMod) || user.IsSuperMod
|
|
|
|
canComment := !blockedInv && user.Perms.CreateProfileReply
|
|
|
|
showComments := c.PrivacyCommentsShow(puser, user)
|
|
|
|
if !showComments {
|
|
|
|
canComment = false
|
|
|
|
}
|
|
|
|
if !c.PrivacyAllowMessage(puser, user) {
|
|
|
|
canMessage = false
|
|
|
|
}
|
2019-10-19 10:33:59 +00:00
|
|
|
|
2022-02-21 03:32:53 +00:00
|
|
|
ppage := c.ProfilePage{h, reList, *puser, currentScore, nextScore, perc, blocked, canMessage, canComment, showComments}
|
|
|
|
return renderTemplate("profile", w, r, h, ppage)
|
2018-02-26 09:07:00 +00:00
|
|
|
}
|