212 lines
6.5 KiB
Go
212 lines
6.5 KiB
Go
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
||
|
|
|
||
|
|
package handler
|
||
|
|
|
||
|
|
import (
|
||
|
|
"errors"
|
||
|
|
"log/slog"
|
||
|
|
"net/http"
|
||
|
|
"strings"
|
||
|
|
|
||
|
|
"kode.naiv.no/olemd/favoritter/internal/image"
|
||
|
|
"kode.naiv.no/olemd/favoritter/internal/middleware"
|
||
|
|
"kode.naiv.no/olemd/favoritter/internal/model"
|
||
|
|
"kode.naiv.no/olemd/favoritter/internal/render"
|
||
|
|
"kode.naiv.no/olemd/favoritter/internal/store"
|
||
|
|
)
|
||
|
|
|
||
|
|
// handlePublicProfile shows a user's public profile and their public faves.
|
||
|
|
func (h *Handler) handlePublicProfile(w http.ResponseWriter, r *http.Request) {
|
||
|
|
username := r.PathValue("username")
|
||
|
|
|
||
|
|
profileUser, err := h.deps.Users.GetByUsername(username)
|
||
|
|
if err != nil {
|
||
|
|
if errors.Is(err, store.ErrUserNotFound) {
|
||
|
|
http.NotFound(w, r)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
slog.Error("get user error", "error", err)
|
||
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
if profileUser.Disabled {
|
||
|
|
http.NotFound(w, r)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
// Check if the viewing user is the profile owner.
|
||
|
|
viewer := middleware.UserFromContext(r.Context())
|
||
|
|
isOwner := viewer != nil && viewer.ID == profileUser.ID
|
||
|
|
|
||
|
|
page := queryInt(r, "page", 1)
|
||
|
|
offset := (page - 1) * defaultPageSize
|
||
|
|
|
||
|
|
var faves []*model.Fave
|
||
|
|
var total int
|
||
|
|
|
||
|
|
if isOwner {
|
||
|
|
faves, total, err = h.deps.Faves.ListByUser(profileUser.ID, defaultPageSize, offset)
|
||
|
|
} else {
|
||
|
|
faves, total, err = h.deps.Faves.ListPublicByUser(profileUser.ID, defaultPageSize, offset)
|
||
|
|
}
|
||
|
|
if err != nil {
|
||
|
|
slog.Error("list faves error", "error", err)
|
||
|
|
http.Error(w, "Internal error", http.StatusInternalServerError)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
if err := h.deps.Faves.LoadTags(faves); err != nil {
|
||
|
|
slog.Error("load tags error", "error", err)
|
||
|
|
}
|
||
|
|
|
||
|
|
totalPages := (total + defaultPageSize - 1) / defaultPageSize
|
||
|
|
|
||
|
|
h.deps.Renderer.Page(w, r, "profile", render.PageData{
|
||
|
|
Title: profileUser.DisplayNameOrUsername() + " sine favoritter",
|
||
|
|
Data: map[string]any{
|
||
|
|
"ProfileUser": profileUser,
|
||
|
|
"IsOwner": isOwner,
|
||
|
|
"IsLimited": profileUser.ProfileVisibility == "limited" && !isOwner,
|
||
|
|
"Faves": faves,
|
||
|
|
"Page": page,
|
||
|
|
"TotalPages": totalPages,
|
||
|
|
"Total": total,
|
||
|
|
},
|
||
|
|
})
|
||
|
|
}
|
||
|
|
|
||
|
|
// handleSettingsGet shows the user settings page.
|
||
|
|
func (h *Handler) handleSettingsGet(w http.ResponseWriter, r *http.Request) {
|
||
|
|
user := middleware.UserFromContext(r.Context())
|
||
|
|
|
||
|
|
h.deps.Renderer.Page(w, r, "settings", render.PageData{
|
||
|
|
Title: "Innstillinger",
|
||
|
|
Data: map[string]any{
|
||
|
|
"SettingsUser": user,
|
||
|
|
},
|
||
|
|
})
|
||
|
|
}
|
||
|
|
|
||
|
|
// handleSettingsPost processes the settings form.
|
||
|
|
func (h *Handler) handleSettingsPost(w http.ResponseWriter, r *http.Request) {
|
||
|
|
user := middleware.UserFromContext(r.Context())
|
||
|
|
|
||
|
|
if err := r.ParseForm(); err != nil {
|
||
|
|
h.flash(w, r, "settings", "Ugyldig forespørsel.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
displayName := strings.TrimSpace(r.FormValue("display_name"))
|
||
|
|
bio := strings.TrimSpace(r.FormValue("bio"))
|
||
|
|
profileVisibility := r.FormValue("profile_visibility")
|
||
|
|
defaultFavePrivacy := r.FormValue("default_fave_privacy")
|
||
|
|
|
||
|
|
if profileVisibility != "public" && profileVisibility != "limited" {
|
||
|
|
profileVisibility = user.ProfileVisibility
|
||
|
|
}
|
||
|
|
if defaultFavePrivacy != "public" && defaultFavePrivacy != "private" {
|
||
|
|
defaultFavePrivacy = user.DefaultFavePrivacy
|
||
|
|
}
|
||
|
|
|
||
|
|
if err := h.deps.Users.UpdateProfile(user.ID, displayName, bio, profileVisibility, defaultFavePrivacy); err != nil {
|
||
|
|
slog.Error("update profile error", "error", err)
|
||
|
|
h.flash(w, r, "settings", "Noe gikk galt. Prøv igjen.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
h.flash(w, r, "settings", "Innstillingene er lagret.", "success", settingsData(user))
|
||
|
|
}
|
||
|
|
|
||
|
|
// handleSettingsPasswordPost handles password change from the settings page.
|
||
|
|
func (h *Handler) handleSettingsPasswordPost(w http.ResponseWriter, r *http.Request) {
|
||
|
|
user := middleware.UserFromContext(r.Context())
|
||
|
|
|
||
|
|
if err := r.ParseForm(); err != nil {
|
||
|
|
h.flash(w, r, "settings", "Ugyldig forespørsel.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
currentPassword := r.FormValue("current_password")
|
||
|
|
newPassword := r.FormValue("new_password")
|
||
|
|
confirmPassword := r.FormValue("confirm_password")
|
||
|
|
|
||
|
|
// Verify current password.
|
||
|
|
if _, err := h.deps.Users.Authenticate(user.Username, currentPassword); err != nil {
|
||
|
|
h.flash(w, r, "settings", "Nåværende passord er feil.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
if len(newPassword) < 8 {
|
||
|
|
h.flash(w, r, "settings", "Nytt passord må være minst 8 tegn.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
if newPassword != confirmPassword {
|
||
|
|
h.flash(w, r, "settings", "Passordene er ikke like.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
if err := h.deps.Users.UpdatePassword(user.ID, newPassword); err != nil {
|
||
|
|
slog.Error("update password error", "error", err)
|
||
|
|
h.flash(w, r, "settings", "Noe gikk galt. Prøv igjen.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
// Invalidate other sessions, keep current one.
|
||
|
|
if delErr := h.deps.Sessions.DeleteAllForUser(user.ID); delErr != nil {
|
||
|
|
slog.Error("session cleanup error", "error", delErr)
|
||
|
|
}
|
||
|
|
token, err := h.deps.Sessions.Create(user.ID)
|
||
|
|
if err != nil {
|
||
|
|
slog.Error("session create error", "error", err)
|
||
|
|
http.Redirect(w, r, h.deps.Config.BasePath+"/login", http.StatusSeeOther)
|
||
|
|
return
|
||
|
|
}
|
||
|
|
h.setSessionCookie(w, r, token)
|
||
|
|
|
||
|
|
h.flash(w, r, "settings", "Passordet er endret.", "success", settingsData(user))
|
||
|
|
}
|
||
|
|
|
||
|
|
// handleAvatarPost handles avatar upload.
|
||
|
|
func (h *Handler) handleAvatarPost(w http.ResponseWriter, r *http.Request) {
|
||
|
|
user := middleware.UserFromContext(r.Context())
|
||
|
|
|
||
|
|
if err := r.ParseMultipartForm(h.deps.Config.MaxUploadSize); err != nil {
|
||
|
|
h.flash(w, r, "settings", "Filen er for stor.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
file, header, err := r.FormFile("avatar")
|
||
|
|
if err != nil {
|
||
|
|
h.flash(w, r, "settings", "Velg et bilde å laste opp.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
defer file.Close()
|
||
|
|
|
||
|
|
result, err := image.Process(file, header, h.deps.Config.UploadDir)
|
||
|
|
if err != nil {
|
||
|
|
slog.Error("avatar process error", "error", err)
|
||
|
|
h.flash(w, r, "settings", "Kunne ikke behandle bildet. Sjekk at filen er et gyldig bilde.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
// Delete old avatar if there was one.
|
||
|
|
if user.AvatarPath != "" {
|
||
|
|
if delErr := image.Delete(h.deps.Config.UploadDir, user.AvatarPath); delErr != nil {
|
||
|
|
slog.Error("avatar delete error", "error", delErr)
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if err := h.deps.Users.UpdateAvatar(user.ID, result.Filename); err != nil {
|
||
|
|
slog.Error("update avatar error", "error", err)
|
||
|
|
h.flash(w, r, "settings", "Noe gikk galt. Prøv igjen.", "error", settingsData(user))
|
||
|
|
return
|
||
|
|
}
|
||
|
|
|
||
|
|
h.flash(w, r, "settings", "Profilbildet er oppdatert.", "success", settingsData(user))
|
||
|
|
}
|
||
|
|
|
||
|
|
func settingsData(user *model.User) map[string]any {
|
||
|
|
return map[string]any{"SettingsUser": user}
|
||
|
|
}
|