Add profile pic and correct usage of banner link
This commit is contained in:
parent
dbddff6e55
commit
b2a8578c72
@ -46,7 +46,7 @@ func GenerateAtomFeed(c *Config, db *DB) (*string, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(article.BannerLink) > 0 {
|
if len(article.BannerLink) > 0 {
|
||||||
linkID := entry.AddLink(atom.NewLink(article.BannerLink))
|
linkID := entry.AddLink(atom.NewLink(c.Domain + "/image/serve/" + article.BannerLink))
|
||||||
entry.Links[linkID].Rel = "enclosure"
|
entry.Links[linkID].Rel = "enclosure"
|
||||||
entry.Links[linkID].Type = "image/webp"
|
entry.Links[linkID].Type = "image/webp"
|
||||||
}
|
}
|
||||||
@ -55,7 +55,8 @@ func GenerateAtomFeed(c *Config, db *DB) (*string, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("error getting user user info for Atom feed: %v", err)
|
return nil, fmt.Errorf("error getting user user info for Atom feed: %v", err)
|
||||||
}
|
}
|
||||||
entry.AddAuthor(atom.NewPerson(user.FirstName + " " + user.LastName))
|
authorID := entry.AddAuthor(atom.NewPerson(user.FirstName + " " + user.LastName))
|
||||||
|
entry.Authors[authorID].URI = c.Domain + "/image/serve/" + user.ProfilePicLink
|
||||||
|
|
||||||
tags, err := db.GetArticleTags(article.ID)
|
tags, err := db.GetArticleTags(article.ID)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -149,11 +149,11 @@ func (db *DB) AddUser(c *Config, u *User, pass string) (int64, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
query := `
|
query := `
|
||||||
INSERT INTO users (username, password, first_name, last_name, email, role)
|
INSERT INTO users (username, password, first_name, last_name, email, profile_pic_link, role)
|
||||||
VALUES (?, ?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?, ?)
|
||||||
`
|
`
|
||||||
|
|
||||||
result, err := db.Exec(query, u.UserName, string(hashedPass), aesFirstName, aesLastName, aesEmail, u.Role)
|
result, err := db.Exec(query, u.UserName, string(hashedPass), aesFirstName, aesLastName, aesEmail, u.ProfilePicLink, u.Role)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return 0, fmt.Errorf("error inserting new user %v into DB: %v", u.UserName, err)
|
return 0, fmt.Errorf("error inserting new user %v into DB: %v", u.UserName, err)
|
||||||
}
|
}
|
||||||
@ -253,13 +253,13 @@ func (db *DB) GetUser(c *Config, id int64) (*User, error) {
|
|||||||
|
|
||||||
user := new(User)
|
user := new(User)
|
||||||
query := `
|
query := `
|
||||||
SELECT id, username, first_name, last_name, email, role
|
SELECT id, username, first_name, last_name, email, profile_pic_link, role
|
||||||
FROM users
|
FROM users
|
||||||
WHERE id = ?
|
WHERE id = ?
|
||||||
`
|
`
|
||||||
|
|
||||||
row := db.QueryRow(query, id)
|
row := db.QueryRow(query, id)
|
||||||
if err := row.Scan(&user.ID, &user.UserName, &aesFirstName, &aesLastName, &aesEmail, &user.Role); err != nil {
|
if err := row.Scan(&user.ID, &user.UserName, &aesFirstName, &aesLastName, &aesEmail, &user.ProfilePicLink, &user.Role); err != nil {
|
||||||
return nil, fmt.Errorf("error reading user information: %v", err)
|
return nil, fmt.Errorf("error reading user information: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -281,10 +281,10 @@ func (db *DB) GetUser(c *Config, id int64) (*User, error) {
|
|||||||
return user, nil
|
return user, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *DB) UpdateOwnUserAttributes(c *Config, id int64, userName, firstName, lastName, email, oldPass, newPass string) error {
|
func (db *DB) UpdateOwnUserAttributes(c *Config, id int64, userName, firstName, lastName, email, profilePicLink, oldPass, newPass string) error {
|
||||||
var err error
|
var err error
|
||||||
tx := new(Tx)
|
tx := new(Tx)
|
||||||
passwordEmpty := len(newPass) > 0
|
passwordEmpty := len(newPass) == 0
|
||||||
|
|
||||||
for i := 0; i < TxMaxRetries; i++ {
|
for i := 0; i < TxMaxRetries; i++ {
|
||||||
err := func() error {
|
err := func() error {
|
||||||
@ -293,6 +293,7 @@ func (db *DB) UpdateOwnUserAttributes(c *Config, id int64, userName, firstName,
|
|||||||
return fmt.Errorf("error starting transaction: %v", err)
|
return fmt.Errorf("error starting transaction: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fmt.Println(len(newPass), passwordEmpty)
|
||||||
if !passwordEmpty {
|
if !passwordEmpty {
|
||||||
if err = tx.ChangePassword(id, oldPass, newPass); err != nil {
|
if err = tx.ChangePassword(id, oldPass, newPass); err != nil {
|
||||||
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
||||||
@ -326,11 +327,13 @@ func (db *DB) UpdateOwnUserAttributes(c *Config, id int64, userName, firstName,
|
|||||||
return fmt.Errorf("error encrypting email: %v", err)
|
return fmt.Errorf("error encrypting email: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fmt.Println("profilePicLink:", profilePicLink)
|
||||||
if err = tx.UpdateAttributes(
|
if err = tx.UpdateAttributes(
|
||||||
&Attribute{Table: "users", ID: id, AttName: "username", Value: userName},
|
&Attribute{Table: "users", ID: id, AttName: "username", Value: userName},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "first_name", Value: aesFirstName},
|
&Attribute{Table: "users", ID: id, AttName: "first_name", Value: aesFirstName},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "last_name", Value: aesLastName},
|
&Attribute{Table: "users", ID: id, AttName: "last_name", Value: aesLastName},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "email", Value: aesEmail},
|
&Attribute{Table: "users", ID: id, AttName: "email", Value: aesEmail},
|
||||||
|
&Attribute{Table: "users", ID: id, AttName: "profile_pic_link", Value: profilePicLink},
|
||||||
); err != nil {
|
); err != nil {
|
||||||
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
||||||
log.Fatalf("transaction error: %v, rollback error: %v", err, rollbackErr)
|
log.Fatalf("transaction error: %v, rollback error: %v", err, rollbackErr)
|
||||||
@ -360,8 +363,8 @@ func (db *DB) AddFirstUser(c *Config, u *User, pass string) (int64, error) {
|
|||||||
txOptions := &sql.TxOptions{Isolation: sql.LevelSerializable}
|
txOptions := &sql.TxOptions{Isolation: sql.LevelSerializable}
|
||||||
selectQuery := "SELECT COUNT(*) FROM users"
|
selectQuery := "SELECT COUNT(*) FROM users"
|
||||||
insertQuery := `
|
insertQuery := `
|
||||||
INSERT INTO users (username, password, first_name, last_name, email, role)
|
INSERT INTO users (username, password, first_name, last_name, email, profile_pic_link, role)
|
||||||
VALUES (?, ?, ?, ?, ?, ?)
|
VALUES (?, ?, ?, ?, ?, ?, ?)
|
||||||
`
|
`
|
||||||
|
|
||||||
for i := 0; i < TxMaxRetries; i++ {
|
for i := 0; i < TxMaxRetries; i++ {
|
||||||
@ -416,7 +419,7 @@ func (db *DB) AddFirstUser(c *Config, u *User, pass string) (int64, error) {
|
|||||||
return 0, fmt.Errorf("error encrypting email: %v", err)
|
return 0, fmt.Errorf("error encrypting email: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
result, err := tx.Exec(insertQuery, u.UserName, string(hashedPass), aesFirstName, aesLastName, aesEmail, u.Role)
|
result, err := tx.Exec(insertQuery, u.UserName, string(hashedPass), aesFirstName, aesLastName, aesEmail, u.ProfilePicLink, u.Role)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
||||||
log.Fatalf("transaction error: %v, rollback error: %v", err, rollbackErr)
|
log.Fatalf("transaction error: %v, rollback error: %v", err, rollbackErr)
|
||||||
@ -451,7 +454,7 @@ func (db *DB) GetAllUsers(c *Config) (map[int64]*User, error) {
|
|||||||
var aesFirstName, aesLastName, aesEmail string
|
var aesFirstName, aesLastName, aesEmail string
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
query := "SELECT id, username, first_name, last_name, email, role FROM users"
|
query := "SELECT id, username, first_name, last_name, email, profile_pic_link, role FROM users"
|
||||||
|
|
||||||
rows, err := db.Query(query)
|
rows, err := db.Query(query)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -461,7 +464,7 @@ func (db *DB) GetAllUsers(c *Config) (map[int64]*User, error) {
|
|||||||
users := make(map[int64]*User, 0)
|
users := make(map[int64]*User, 0)
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
user := new(User)
|
user := new(User)
|
||||||
if err = rows.Scan(&user.ID, &user.UserName, &aesFirstName, &aesLastName, &aesEmail, &user.Role); err != nil {
|
if err = rows.Scan(&user.ID, &user.UserName, &aesFirstName, &aesLastName, &aesEmail, &user.ProfilePicLink, &user.Role); err != nil {
|
||||||
return nil, fmt.Errorf("error getting user info: %v", err)
|
return nil, fmt.Errorf("error getting user info: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -506,10 +509,10 @@ func (tx *Tx) SetPassword(id int64, newPass string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *DB) UpdateUserAttributes(c *Config, id int64, userName, firstName, lastName, email, newPass string, role int) error {
|
func (db *DB) UpdateUserAttributes(c *Config, id int64, userName, firstName, lastName, email, profilePicLink, newPass string, role int) error {
|
||||||
var err error
|
var err error
|
||||||
tx := new(Tx)
|
tx := new(Tx)
|
||||||
passwordEmpty := len(newPass) > 0
|
passwordEmpty := len(newPass) == 0
|
||||||
|
|
||||||
for i := 0; i < TxMaxRetries; i++ {
|
for i := 0; i < TxMaxRetries; i++ {
|
||||||
err := func() error {
|
err := func() error {
|
||||||
@ -556,6 +559,7 @@ func (db *DB) UpdateUserAttributes(c *Config, id int64, userName, firstName, las
|
|||||||
&Attribute{Table: "users", ID: id, AttName: "first_name", Value: aesFirstName},
|
&Attribute{Table: "users", ID: id, AttName: "first_name", Value: aesFirstName},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "last_name", Value: aesLastName},
|
&Attribute{Table: "users", ID: id, AttName: "last_name", Value: aesLastName},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "email", Value: aesEmail},
|
&Attribute{Table: "users", ID: id, AttName: "email", Value: aesEmail},
|
||||||
|
&Attribute{Table: "users", ID: id, AttName: "profile_pic_link", Value: profilePicLink},
|
||||||
&Attribute{Table: "users", ID: id, AttName: "role", Value: role},
|
&Attribute{Table: "users", ID: id, AttName: "role", Value: role},
|
||||||
); err != nil {
|
); err != nil {
|
||||||
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
if rollbackErr := tx.Rollback(); rollbackErr != nil {
|
||||||
|
@ -7,7 +7,6 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
b "streifling.com/jason/cpolis/cmd/backend"
|
b "streifling.com/jason/cpolis/cmd/backend"
|
||||||
@ -24,7 +23,7 @@ type EditorHTMLData struct {
|
|||||||
Action string
|
Action string
|
||||||
ActionTitle string
|
ActionTitle string
|
||||||
ActionButton string
|
ActionButton string
|
||||||
BannerImage string
|
Image string
|
||||||
HTMLContent template.HTML
|
HTMLContent template.HTML
|
||||||
Article *b.Article
|
Article *b.Article
|
||||||
Tags []*b.Tag
|
Tags []*b.Tag
|
||||||
@ -80,7 +79,7 @@ func SubmitArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
|
|
||||||
article := &b.Article{
|
article := &b.Article{
|
||||||
Title: r.PostFormValue("article-title"),
|
Title: r.PostFormValue("article-title"),
|
||||||
BannerLink: c.Domain + "/image/serve/" + r.PostFormValue("article-banner-url"),
|
BannerLink: r.PostFormValue("article-banner-url"),
|
||||||
Summary: r.PostFormValue("article-summary"),
|
Summary: r.PostFormValue("article-summary"),
|
||||||
Published: false,
|
Published: false,
|
||||||
Rejected: false,
|
Rejected: false,
|
||||||
@ -170,11 +169,6 @@ func ResubmitArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
bannerLink := r.PostFormValue("article-banner-url")
|
|
||||||
if len(bannerLink) != 0 {
|
|
||||||
bannerLink = c.Domain + "/image/serve/" + bannerLink
|
|
||||||
}
|
|
||||||
|
|
||||||
summary := r.PostFormValue("article-summary")
|
summary := r.PostFormValue("article-summary")
|
||||||
if len(summary) == 0 {
|
if len(summary) == 0 {
|
||||||
http.Error(w, "Bitte die Beschreibung eingeben.", http.StatusBadRequest)
|
http.Error(w, "Bitte die Beschreibung eingeben.", http.StatusBadRequest)
|
||||||
@ -196,7 +190,7 @@ func ResubmitArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
|
|
||||||
if err = db.UpdateAttributes(
|
if err = db.UpdateAttributes(
|
||||||
&b.Attribute{Table: "articles", ID: id, AttName: "title", Value: title},
|
&b.Attribute{Table: "articles", ID: id, AttName: "title", Value: title},
|
||||||
&b.Attribute{Table: "articles", ID: id, AttName: "banner_link", Value: bannerLink},
|
&b.Attribute{Table: "articles", ID: id, AttName: "banner_link", Value: r.PostFormValue("article-banner-url")},
|
||||||
&b.Attribute{Table: "articles", ID: id, AttName: "summary", Value: summary},
|
&b.Attribute{Table: "articles", ID: id, AttName: "summary", Value: summary},
|
||||||
&b.Attribute{Table: "articles", ID: id, AttName: "rejected", Value: false},
|
&b.Attribute{Table: "articles", ID: id, AttName: "rejected", Value: false},
|
||||||
&b.Attribute{Table: "articles", ID: id, AttName: "is_in_issue", Value: r.PostFormValue("issue") == "on"},
|
&b.Attribute{Table: "articles", ID: id, AttName: "is_in_issue", Value: r.PostFormValue("issue") == "on"},
|
||||||
@ -341,8 +335,7 @@ func ReviewRejectedArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.Handler
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
imgURL := strings.Split(data.Article.BannerLink, "/")
|
data.Image = data.Article.BannerLink
|
||||||
data.BannerImage = imgURL[len(imgURL)-1]
|
|
||||||
|
|
||||||
articleAbsName := fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md")
|
articleAbsName := fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md")
|
||||||
content, err := os.ReadFile(articleAbsName)
|
content, err := os.ReadFile(articleAbsName)
|
||||||
@ -612,8 +605,7 @@ func ReviewArticle(c *b.Config, db *b.DB, s *b.CookieStore, action, title, butto
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
imgURL := strings.Split(article.BannerLink, "/")
|
data.Image = article.BannerLink
|
||||||
data.BannerImage = imgURL[len(imgURL)-1]
|
|
||||||
|
|
||||||
data.Article.Summary, err = b.ConvertToPlain(article.Summary)
|
data.Article.Summary, err = b.ConvertToPlain(article.Summary)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -789,8 +781,7 @@ func EditArticle(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
imgURL := strings.Split(data.Article.BannerLink, "/")
|
data.Image = data.Article.BannerLink
|
||||||
data.BannerImage = imgURL[len(imgURL)-1]
|
|
||||||
|
|
||||||
content, err := os.ReadFile(fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md"))
|
content, err := os.ReadFile(fmt.Sprint(c.ArticleDir, "/", data.Article.ID, ".md"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -9,7 +9,7 @@ import (
|
|||||||
b "streifling.com/jason/cpolis/cmd/backend"
|
b "streifling.com/jason/cpolis/cmd/backend"
|
||||||
)
|
)
|
||||||
|
|
||||||
func UploadImage(c *b.Config, s *b.CookieStore) http.HandlerFunc {
|
func UploadEasyMDEImage(c *b.Config, s *b.CookieStore) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
if _, err := GetSession(w, r, c, s); err != nil {
|
if _, err := GetSession(w, r, c, s); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
@ -42,7 +42,7 @@ func UploadImage(c *b.Config, s *b.CookieStore) http.HandlerFunc {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func UploadBanner(c *b.Config, s *b.CookieStore, fileKey, htmlFile, htmlTemplate string) http.HandlerFunc {
|
func UploadImage(c *b.Config, s *b.CookieStore, fileKey, htmlFile, htmlTemplate string) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
if _, err := GetSession(w, r, c, s); err != nil {
|
if _, err := GetSession(w, r, c, s); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
@ -69,8 +69,8 @@ func UploadBanner(c *b.Config, s *b.CookieStore, fileKey, htmlFile, htmlTemplate
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
data := new(struct{ BannerImage string })
|
data := new(struct{ Image string })
|
||||||
data.BannerImage = filename
|
data.Image = filename
|
||||||
|
|
||||||
tmpl, err := template.ParseFiles(c.WebDir + "/templates/" + htmlFile)
|
tmpl, err := template.ParseFiles(c.WebDir + "/templates/" + htmlFile)
|
||||||
if err = template.Must(tmpl, err).ExecuteTemplate(w, htmlTemplate, data); err != nil {
|
if err = template.Must(tmpl, err).ExecuteTemplate(w, htmlTemplate, data); err != nil {
|
||||||
|
@ -57,15 +57,21 @@ func HomePage(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
data := new(struct {
|
data := new(struct {
|
||||||
|
*UserHTMLData
|
||||||
Version string
|
Version string
|
||||||
Role int
|
|
||||||
})
|
})
|
||||||
|
data.UserHTMLData = &UserHTMLData{User: new(b.User)}
|
||||||
data.Version = c.Version
|
data.Version = c.Version
|
||||||
|
|
||||||
files := make([]string, 2)
|
files := make([]string, 2)
|
||||||
files[0] = c.WebDir + "/templates/index.html"
|
files[0] = c.WebDir + "/templates/index.html"
|
||||||
if numRows == 0 {
|
if numRows == 0 {
|
||||||
files[1] = c.WebDir + "/templates/first-user.html"
|
data.Role = b.NonExistent
|
||||||
|
data.Title = "Erster Benutzer (Administrator)"
|
||||||
|
data.ButtonText = "Anlegen"
|
||||||
|
data.URL = "/user/add-first"
|
||||||
|
|
||||||
|
files[1] = c.WebDir + "/templates/edit-user.html"
|
||||||
tmpl, err := template.ParseFiles(files...)
|
tmpl, err := template.ParseFiles(files...)
|
||||||
if err = template.Must(tmpl, err).Execute(w, data); err != nil {
|
if err = template.Must(tmpl, err).Execute(w, data); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
@ -79,6 +85,7 @@ func HomePage(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if auth, ok := session.Values["authenticated"].(bool); auth && ok {
|
if auth, ok := session.Values["authenticated"].(bool); auth && ok {
|
||||||
data.Role = session.Values["role"].(int)
|
data.Role = session.Values["role"].(int)
|
||||||
files[1] = c.WebDir + "/templates/hub.html"
|
files[1] = c.WebDir + "/templates/hub.html"
|
||||||
@ -89,6 +96,7 @@ func HomePage(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
data.Role = b.Author
|
||||||
files[1] = c.WebDir + "/templates/login.html"
|
files[1] = c.WebDir + "/templates/login.html"
|
||||||
tmpl, err := template.ParseFiles(files...)
|
tmpl, err := template.ParseFiles(files...)
|
||||||
if err = template.Must(tmpl, err).Execute(w, data); err != nil {
|
if err = template.Must(tmpl, err).Execute(w, data); err != nil {
|
||||||
|
@ -10,6 +10,14 @@ import (
|
|||||||
b "streifling.com/jason/cpolis/cmd/backend"
|
b "streifling.com/jason/cpolis/cmd/backend"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type UserHTMLData struct {
|
||||||
|
*b.User
|
||||||
|
Title string
|
||||||
|
ButtonText string
|
||||||
|
URL string
|
||||||
|
Image string
|
||||||
|
}
|
||||||
|
|
||||||
func checkUserStrings(user *b.User) (string, int, bool) {
|
func checkUserStrings(user *b.User) (string, int, bool) {
|
||||||
userLen := 63 // max value for utf-8 at 255 bytes
|
userLen := 63 // max value for utf-8 at 255 bytes
|
||||||
nameLen := 56 // max value when aes encrypting utf-8 at up to 255 bytes
|
nameLen := 56 // max value when aes encrypting utf-8 at up to 255 bytes
|
||||||
@ -33,8 +41,15 @@ func CreateUser(c *b.Config, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
tmpl, err := template.ParseFiles(c.WebDir + "/templates/add-user.html")
|
data := &UserHTMLData{
|
||||||
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", nil); err != nil {
|
User: &b.User{Role: b.Author},
|
||||||
|
Title: "Neuer Benutzer",
|
||||||
|
ButtonText: "Anlegen",
|
||||||
|
URL: "/user/add",
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpl, err := template.ParseFiles(c.WebDir + "/templates/edit-user.html")
|
||||||
|
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
@ -52,10 +67,11 @@ func AddUser(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
user := &b.User{
|
user := &b.User{
|
||||||
UserName: r.PostFormValue("username"),
|
UserName: r.PostFormValue("username"),
|
||||||
FirstName: r.PostFormValue("first-name"),
|
FirstName: r.PostFormValue("first-name"),
|
||||||
LastName: r.PostFormValue("last-name"),
|
LastName: r.PostFormValue("last-name"),
|
||||||
Email: r.PostFormValue("email"),
|
Email: r.PostFormValue("email"),
|
||||||
|
ProfilePicLink: r.PostFormValue("profile-pic-url"),
|
||||||
}
|
}
|
||||||
pass := r.PostFormValue("password")
|
pass := r.PostFormValue("password")
|
||||||
pass2 := r.PostFormValue("password2")
|
pass2 := r.PostFormValue("password2")
|
||||||
@ -136,8 +152,16 @@ func EditSelf(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
tmpl, err := template.ParseFiles(c.WebDir + "/templates/edit-self.html")
|
data := &UserHTMLData{
|
||||||
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", user); err != nil {
|
User: user,
|
||||||
|
Title: "Mein Profil bearbeiten",
|
||||||
|
ButtonText: "Übernehmen",
|
||||||
|
URL: "/user/update/self",
|
||||||
|
Image: user.ProfilePicLink,
|
||||||
|
}
|
||||||
|
|
||||||
|
tmpl, err := template.ParseFiles(c.WebDir + "/templates/edit-user.html")
|
||||||
|
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
@ -155,11 +179,12 @@ func UpdateSelf(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
user := &b.User{
|
user := &b.User{
|
||||||
ID: session.Values["id"].(int64),
|
ID: session.Values["id"].(int64),
|
||||||
UserName: r.PostFormValue("username"),
|
UserName: r.PostFormValue("username"),
|
||||||
FirstName: r.PostFormValue("first-name"),
|
FirstName: r.PostFormValue("first-name"),
|
||||||
LastName: r.PostFormValue("last-name"),
|
LastName: r.PostFormValue("last-name"),
|
||||||
Email: r.PostFormValue("email"),
|
Email: r.PostFormValue("email"),
|
||||||
|
ProfilePicLink: r.PostFormValue("profile-pic-url"),
|
||||||
}
|
}
|
||||||
|
|
||||||
oldPass := r.PostFormValue("old-password")
|
oldPass := r.PostFormValue("old-password")
|
||||||
@ -202,7 +227,7 @@ func UpdateSelf(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = db.UpdateOwnUserAttributes(c, user.ID, user.UserName, user.FirstName, user.LastName, user.Email, oldPass, newPass); err != nil {
|
if err = db.UpdateOwnUserAttributes(c, user.ID, user.UserName, user.FirstName, user.LastName, user.Email, user.ProfilePicLink, oldPass, newPass); err != nil {
|
||||||
log.Println("error: user:", user.ID, err)
|
log.Println("error: user:", user.ID, err)
|
||||||
http.Error(w, "Benutzerdaten konnten nicht aktualisiert werden.", http.StatusInternalServerError)
|
http.Error(w, "Benutzerdaten konnten nicht aktualisiert werden.", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
@ -225,11 +250,12 @@ func AddFirstUser(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
var err error
|
var err error
|
||||||
user := &b.User{
|
user := &b.User{
|
||||||
UserName: r.PostFormValue("username"),
|
UserName: r.PostFormValue("username"),
|
||||||
FirstName: r.PostFormValue("first-name"),
|
FirstName: r.PostFormValue("first-name"),
|
||||||
LastName: r.PostFormValue("last-name"),
|
LastName: r.PostFormValue("last-name"),
|
||||||
Email: r.PostFormValue("email"),
|
Email: r.PostFormValue("email"),
|
||||||
Role: b.Admin,
|
ProfilePicLink: r.PostFormValue("profile-pic-url"),
|
||||||
|
Role: b.Admin,
|
||||||
}
|
}
|
||||||
pass := r.PostFormValue("password")
|
pass := r.PostFormValue("password")
|
||||||
pass2 := r.PostFormValue("password2")
|
pass2 := r.PostFormValue("password2")
|
||||||
@ -345,8 +371,16 @@ func EditUser(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data := &UserHTMLData{
|
||||||
|
User: user,
|
||||||
|
Title: "Profil von " + user.FirstName + " " + user.LastName + " bearbeiten",
|
||||||
|
ButtonText: "Übernehmen",
|
||||||
|
URL: fmt.Sprint("/user/update/", user.ID),
|
||||||
|
Image: user.ProfilePicLink,
|
||||||
|
}
|
||||||
|
|
||||||
tmpl, err := template.ParseFiles(c.WebDir + "/templates/edit-user.html")
|
tmpl, err := template.ParseFiles(c.WebDir + "/templates/edit-user.html")
|
||||||
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", user); err != nil {
|
if err = template.Must(tmpl, err).ExecuteTemplate(w, "page-content", data); err != nil {
|
||||||
log.Println(err)
|
log.Println(err)
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
@ -402,6 +436,8 @@ func UpdateUser(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
user.ProfilePicLink = r.PostFormValue("profile-pic-url")
|
||||||
|
|
||||||
newPass := r.PostFormValue("password")
|
newPass := r.PostFormValue("password")
|
||||||
newPass2 := r.PostFormValue("password2")
|
newPass2 := r.PostFormValue("password2")
|
||||||
if newPass != newPass2 {
|
if newPass != newPass2 {
|
||||||
@ -420,7 +456,7 @@ func UpdateUser(c *b.Config, db *b.DB, s *b.CookieStore) http.HandlerFunc {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = db.UpdateUserAttributes(c, user.ID, user.UserName, user.FirstName, user.LastName, user.Email, newPass, user.Role); err != nil {
|
if err = db.UpdateUserAttributes(c, user.ID, user.UserName, user.FirstName, user.LastName, user.Email, user.ProfilePicLink, newPass, user.Role); err != nil {
|
||||||
log.Println("error: user:", user.ID, err)
|
log.Println("error: user:", user.ID, err)
|
||||||
http.Error(w, "Benutzerdaten konnten nicht aktualisiert werden.", http.StatusInternalServerError)
|
http.Error(w, "Benutzerdaten konnten nicht aktualisiert werden.", http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
|
@ -83,10 +83,10 @@ func main() {
|
|||||||
|
|
||||||
mux.HandleFunc("POST /article/resubmit/{id}", f.ResubmitArticle(config, db, store))
|
mux.HandleFunc("POST /article/resubmit/{id}", f.ResubmitArticle(config, db, store))
|
||||||
mux.HandleFunc("POST /article/submit", f.SubmitArticle(config, db, store))
|
mux.HandleFunc("POST /article/submit", f.SubmitArticle(config, db, store))
|
||||||
mux.HandleFunc("POST /article/upload-banner", f.UploadBanner(config, store, "article-banner", "editor.html", "article-banner-template"))
|
mux.HandleFunc("POST /article/upload-banner", f.UploadImage(config, store, "article-banner", "editor.html", "article-banner-template"))
|
||||||
mux.HandleFunc("POST /article/upload-image", f.UploadImage(config, store))
|
mux.HandleFunc("POST /article/upload-image", f.UploadEasyMDEImage(config, store))
|
||||||
mux.HandleFunc("POST /issue/publish", f.PublishLatestIssue(config, db, store))
|
mux.HandleFunc("POST /issue/publish", f.PublishLatestIssue(config, db, store))
|
||||||
mux.HandleFunc("POST /issue/upload-banner", f.UploadBanner(config, store, "issue-banner", "current-issue.html", "issue-banner-template"))
|
mux.HandleFunc("POST /issue/upload-banner", f.UploadImage(config, store, "issue-banner", "current-issue.html", "issue-banner-template"))
|
||||||
mux.HandleFunc("POST /login", f.Login(config, db, store))
|
mux.HandleFunc("POST /login", f.Login(config, db, store))
|
||||||
mux.HandleFunc("POST /pdf/upload", f.UploadPDF(config, store))
|
mux.HandleFunc("POST /pdf/upload", f.UploadPDF(config, store))
|
||||||
mux.HandleFunc("POST /tag/add", f.AddTag(config, db, store))
|
mux.HandleFunc("POST /tag/add", f.AddTag(config, db, store))
|
||||||
@ -94,6 +94,7 @@ func main() {
|
|||||||
mux.HandleFunc("POST /user/add-first", f.AddFirstUser(config, db, store))
|
mux.HandleFunc("POST /user/add-first", f.AddFirstUser(config, db, store))
|
||||||
mux.HandleFunc("POST /user/update/{id}", f.UpdateUser(config, db, store))
|
mux.HandleFunc("POST /user/update/{id}", f.UpdateUser(config, db, store))
|
||||||
mux.HandleFunc("POST /user/update/self", f.UpdateSelf(config, db, store))
|
mux.HandleFunc("POST /user/update/self", f.UpdateSelf(config, db, store))
|
||||||
|
mux.HandleFunc("POST /user/upload-profile-pic", f.UploadImage(config, store, "upload-profile-pic", "edit-user.html", "profile-pic-template"))
|
||||||
|
|
||||||
log.Fatalln(http.ListenAndServe(config.Port, mux))
|
log.Fatalln(http.ListenAndServe(config.Port, mux))
|
||||||
}
|
}
|
||||||
|
@ -11,7 +11,7 @@ CREATE TABLE users (
|
|||||||
first_name VARCHAR(255) NOT NULL,
|
first_name VARCHAR(255) NOT NULL,
|
||||||
last_name VARCHAR(255) NOT NULL,
|
last_name VARCHAR(255) NOT NULL,
|
||||||
email VARCHAR(255) NOT NULL,
|
email VARCHAR(255) NOT NULL,
|
||||||
-- profile_pic_link VARCHAR(255) NOT NULL,
|
profile_pic_link VARCHAR(255),
|
||||||
role INT NOT NULL,
|
role INT NOT NULL,
|
||||||
PRIMARY KEY (id)
|
PRIMARY KEY (id)
|
||||||
);
|
);
|
||||||
@ -26,7 +26,7 @@ CREATE TABLE articles (
|
|||||||
id INT AUTO_INCREMENT,
|
id INT AUTO_INCREMENT,
|
||||||
title VARCHAR(255) NOT NULL,
|
title VARCHAR(255) NOT NULL,
|
||||||
created TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
created TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
|
||||||
banner_link VARCHAR(255) NOT NULL,
|
banner_link VARCHAR(255),
|
||||||
summary TEXT NOT NULL,
|
summary TEXT NOT NULL,
|
||||||
published BOOL NOT NULL,
|
published BOOL NOT NULL,
|
||||||
rejected BOOL NOT NULL,
|
rejected BOOL NOT NULL,
|
||||||
|
@ -24,7 +24,7 @@ textarea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.btn-area {
|
.btn-area {
|
||||||
@apply grid grid-cols-1 md:grid-cols-2 gap-x-4 gap-y-1 mt-4;
|
@apply grid grid-cols-1 md:grid-cols-2 gap-x-4 gap-y-1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn-area-3 {
|
.btn-area-3 {
|
||||||
@ -40,33 +40,33 @@ textarea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.EasyMDEContainer .CodeMirror {
|
.EasyMDEContainer .CodeMirror {
|
||||||
@apply bg-slate-50 dark:bg-slate-950 border-slate-200 dark:border-slate-800 text-slate-900 dark:text-slate-100
|
@apply bg-slate-50 dark:bg-slate-950 border-slate-200 dark:border-slate-800 text-slate-900 dark:text-slate-100;
|
||||||
}
|
}
|
||||||
|
|
||||||
.EasyMDEContainer .cm-s-easymde .CodeMirror-cursor {
|
.EasyMDEContainer .cm-s-easymde .CodeMirror-cursor {
|
||||||
@apply border-slate-900 dark:border-slate-100
|
@apply border-slate-900 dark:border-slate-100;
|
||||||
}
|
}
|
||||||
|
|
||||||
.EasyMDEContainer .editor-toolbar > * {
|
.EasyMDEContainer .editor-toolbar > * {
|
||||||
@apply text-slate-900 dark:text-slate-100
|
@apply text-slate-900 dark:text-slate-100;
|
||||||
}
|
}
|
||||||
|
|
||||||
.EasyMDEContainer .editor-toolbar > .active, .editor-toolbar > button:hover, .editor-preview pre, .cm-s-easymde .cm-comment {
|
.EasyMDEContainer .editor-toolbar > .active, .editor-toolbar > button:hover, .editor-preview pre, .cm-s-easymde .cm-comment {
|
||||||
@apply bg-slate-100 dark:bg-slate-900
|
@apply bg-slate-100 dark:bg-slate-900;
|
||||||
}
|
}
|
||||||
|
|
||||||
.EasyMDEContainer .CodeMirror-fullscreen {
|
.EasyMDEContainer .CodeMirror-fullscreen {
|
||||||
@apply bg-slate-50 dark:bg-slate-950
|
@apply bg-slate-50 dark:bg-slate-950;
|
||||||
}
|
}
|
||||||
|
|
||||||
.editor-toolbar {
|
.editor-toolbar {
|
||||||
@apply border border-slate-200 dark:border-slate-800
|
@apply border border-slate-200 dark:border-slate-800;
|
||||||
}
|
}
|
||||||
|
|
||||||
.editor-toolbar.fullscreen {
|
.editor-toolbar.fullscreen {
|
||||||
@apply bg-slate-50 dark:bg-slate-950
|
@apply bg-slate-50 dark:bg-slate-950;
|
||||||
}
|
}
|
||||||
|
|
||||||
.editor-preview {
|
.editor-preview {
|
||||||
@apply bg-slate-50 dark:bg-slate-950
|
@apply bg-slate-50 dark:bg-slate-950;
|
||||||
}
|
}
|
||||||
|
@ -1,66 +0,0 @@
|
|||||||
{{define "page-content"}}
|
|
||||||
<h2>Neuer Benutzer</h2>
|
|
||||||
|
|
||||||
<form>
|
|
||||||
<div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
|
|
||||||
<div>
|
|
||||||
<label for="username">Benutzername</label>
|
|
||||||
<input class="w-full" required name="username" type="text" value="{{.UserName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password">Passwort</label>
|
|
||||||
<input class="w-full" required name="password" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password2">Passwort wiederholen</label>
|
|
||||||
<input class="w-full" required name="password2" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="first-name">Vorname</label>
|
|
||||||
<input class="w-full" required name="first-name" type="text" value="{{.FirstName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="last-name">Nachname</label>
|
|
||||||
<input class="w-full" required name="last-name" type="text" value="{{.LastName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email">Emailadresse</label>
|
|
||||||
<input class="w-full" required name="email" type="text" value="{{.Email}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email2">Emailadresse wiederholen</label>
|
|
||||||
<input class="w-full" required name="email2" type="text" value="{{.Email}}" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex flex-wrap gap-4">
|
|
||||||
<div>
|
|
||||||
<input required id="author" name="role" type="radio" value="3" {{if eq .Role 3 }}checked{{end}} />
|
|
||||||
<label for="author">Autor</label>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<input required id="editor" name="role" type="radio" value="2" {{if eq .Role 2 }}checked{{end}} />
|
|
||||||
<label for="editor">Redakteur</label>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<input required id="publisher" name="role" type="radio" value="1" {{if eq .Role 1 }}checked{{end}} />
|
|
||||||
<label for="publisher">Herausgeber</label>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<input required id="admin" name="role" type="radio" value="0" {{if eq .Role 0 }}checked{{end}} />
|
|
||||||
<label for="admin">Administrator</label>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="btn-area">
|
|
||||||
<input class="action-btn" type="submit" value="Anlegen" hx-post="/user/add" hx-target="#page-content" />
|
|
||||||
<button class="btn" hx-get="/hub" hx-target="#page-content">Abbrechen</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
{{end}}
|
|
@ -79,8 +79,8 @@
|
|||||||
{{end}}
|
{{end}}
|
||||||
|
|
||||||
{{define "issue-banner-template"}}
|
{{define "issue-banner-template"}}
|
||||||
<div class="w-full" id="issue-banner-container">
|
<div id="issue-banner-container">
|
||||||
<img src="data:image/webp;base64,{{.BannerImage}}" alt="Banner Image">
|
<img src="/image/serve/{{.Image}}" alt="" />
|
||||||
<input id="issue-banner-url" name="issue-banner-url" type="hidden" value="{{.URL}}" />
|
<input id="issue-banner-url" name="issue-banner-url" type="hidden" value="{{.Image}}" />
|
||||||
</div>
|
</div>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
@ -1,53 +0,0 @@
|
|||||||
{{define "page-content"}}
|
|
||||||
<h2>Profil bearbeiten</h2>
|
|
||||||
|
|
||||||
<form>
|
|
||||||
<div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
|
|
||||||
<div>
|
|
||||||
<label for="username">Benutzername</label>
|
|
||||||
<input class="w-full" name="username" type="text" value="{{.UserName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="first-name">Vorname</label>
|
|
||||||
<input class="w-full" name="first-name" type="text" value="{{.FirstName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="last-name">Nachname</label>
|
|
||||||
<input class="w-full" name="last-name" type="text" value="{{.LastName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="old-password">Altes Passwort</label>
|
|
||||||
<input class="w-full" name="old-password" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password">Passwort</label>
|
|
||||||
<input class="w-full" name="password" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password2">Passwort wiederholen</label>
|
|
||||||
<input class="w-full" name="password2" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email">Emailadresse</label>
|
|
||||||
<input class="w-full" required name="email" type="text" value="{{.Email}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email2">Emailadresse wiederholen</label>
|
|
||||||
<input class="w-full" required name="email2" type="text" value="{{.Email}}" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="btn-area">
|
|
||||||
<input class="action-btn" type="submit" value="Aktualisieren" hx-post="/user/update/self"
|
|
||||||
hx-target="#page-content" />
|
|
||||||
<button class="btn" hx-get="/hub" hx-target="#page-content">Abbrechen</button>
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
{{end}}
|
|
@ -1,67 +1,90 @@
|
|||||||
{{define "page-content"}}
|
{{define "page-content"}}
|
||||||
<h2>Profil von {{.FirstName}} {{.LastName}} bearbeiten</h2>
|
<h2>{{.Title}}</h2>
|
||||||
|
|
||||||
|
<form class="flex flex-col gap-4" hx-encoding="multipart/form-data">
|
||||||
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4">
|
||||||
|
{{template "profile-pic-template" .}}
|
||||||
|
|
||||||
<form>
|
|
||||||
<div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 gap-4">
|
|
||||||
<div>
|
<div>
|
||||||
<label for="username">Benutzername</label>
|
<label for="username">Benutzername</label>
|
||||||
<input class="w-full" name="username" type="text" value="{{.UserName}}" />
|
<input class="w-full" required name="username" type="text" {{if lt .Role 4}}value="{{.UserName}}" {{end}} />
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="first-name">Vorname</label>
|
|
||||||
<input class="w-full" name="first-name" type="text" value="{{.FirstName}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="last-name">Nachname</label>
|
|
||||||
<input class="w-full" name="last-name" type="text" value="{{.LastName}}" />
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label for="password">Passwort</label>
|
<label for="password">Passwort</label>
|
||||||
<input class="w-full" name="password" placeholder="***" type="password" />
|
<input class="w-full" required name="password" placeholder="***" type="password" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label for="password2">Passwort wiederholen</label>
|
<label for="password2">Passwort wiederholen</label>
|
||||||
<input class="w-full" name="password2" placeholder="***" type="password" />
|
<input class="w-full" required name="password2" placeholder="***" type="password" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<label for="first-name">Vorname</label>
|
||||||
|
<input class="w-full" required name="first-name" type="text" {{if lt .Role 4}}value="{{.FirstName}}"
|
||||||
|
{{end}} />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<label for="last-name">Nachname</label>
|
||||||
|
<input class="w-full" required name="last-name" type="text" {{if lt .Role 4}}value="{{.LastName}}"
|
||||||
|
{{end}} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label for="email">Emailadresse</label>
|
<label for="email">Emailadresse</label>
|
||||||
<input class="w-full" required name="email" type="text" value="{{.Email}}" />
|
<input class="w-full" required name="email" type="text" {{if lt .Role 4}}value="{{.Email}}" {{end}} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label for="email2">Emailadresse wiederholen</label>
|
<label for="email2">Emailadresse wiederholen</label>
|
||||||
<input class="w-full" required name="email2" type="text" value="{{.Email}}" />
|
<input class="w-full" required name="email2" type="text" {{if lt .Role 4}}value="{{.Email}}" {{end}} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{{if lt .Role 4}}
|
||||||
<div class="flex flex-wrap gap-4">
|
<div class="flex flex-wrap gap-4">
|
||||||
<div>
|
<div>
|
||||||
<input required id="author" name="role" type="radio" value="3" {{if eq .Role 3 }}checked{{end}} />
|
<input required id="author" name="role" type="radio" value="3" {{if eq .Role 3}}checked{{end}} />
|
||||||
<label for="author">Autor</label>
|
<label for="author">Autor</label>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<input required id="editor" name="role" type="radio" value="2" {{if eq .Role 2 }}checked{{end}} />
|
<input required id="editor" name="role" type="radio" value="2" {{if eq .Role 2}}checked{{end}} />
|
||||||
<label for="editor">Redakteur</label>
|
<label for="editor">Redakteur</label>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<input required id="publisher" name="role" type="radio" value="1" {{if eq .Role 1 }}checked{{end}} />
|
<input required id="publisher" name="role" type="radio" value="1" {{if eq .Role 1}}checked{{end}} />
|
||||||
<label for="publisher">Herausgeber</label>
|
<label for="publisher">Herausgeber</label>
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<input required id="admin" name="role" type="radio" value="0" {{if eq .Role 0 }}checked{{end}} />
|
<input required id="admin" name="role" type="radio" value="0" {{if eq .Role 0}}checked{{end}} />
|
||||||
<label for="admin">Administrator</label>
|
<label for="admin">Administrator</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
{{end}}
|
||||||
|
|
||||||
<div class="btn-area">
|
<div class="btn-area">
|
||||||
<input class="action-btn" type="submit" value="Aktualisieren" hx-post="/user/update/{{.ID}}"
|
<input class="action-btn" type="submit" value="{{.ButtonText}}" hx-post="{{.URL}}" hx-target="#page-content" />
|
||||||
hx-target="#page-content" />
|
|
||||||
<button class="btn" hx-get="/hub" hx-target="#page-content">Abbrechen</button>
|
<button class="btn" hx-get="/hub" hx-target="#page-content">Abbrechen</button>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
|
||||||
|
{{define "profile-pic-template"}}
|
||||||
|
<div class="flex items-center justify-center row-span-3 self-center" id="profile-pic-container">
|
||||||
|
<label
|
||||||
|
class="bg-slate-200 dark:bg-slate-800 hover:bg-slate-100 dark:hover:bg-slate-900 border border-slate-200 dark:border-slate-800 cursor-pointer flex flex-col h-48 items-center justify-center overflow-hidden rounded-full w-48"
|
||||||
|
for="upload-profile-pic">
|
||||||
|
{{if gt (len .Image) 0}}
|
||||||
|
<img src="/image/serve/{{.Image}}" alt="" />
|
||||||
|
{{else}}
|
||||||
|
<span class="text-2xl">+</span>
|
||||||
|
<span>Profilbild</span>
|
||||||
|
{{end}}
|
||||||
|
</label>
|
||||||
|
<input class="hidden" id="upload-profile-pic" name="upload-profile-pic" type="file"
|
||||||
|
hx-post="/user/upload-profile-pic" hx-swap="outerHTML" hx-target="#profile-pic-container" />
|
||||||
|
<input id="profile-pic-url" name="profile-pic-url" type="hidden" value="{{.Image}}" />
|
||||||
|
</div>
|
||||||
|
{{end}}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{{define "page-content"}}
|
{{define "page-content"}}
|
||||||
<h2>Editor</h2>
|
<h2>Editor</h2>
|
||||||
|
|
||||||
<form id="edit-area" hx-encoding="multipart/form-data">
|
<form class="flex flex-col gap-4" id="edit-area" hx-encoding="multipart/form-data">
|
||||||
<div class="flex flex-col gap-y-1">
|
<div class="flex flex-col gap-y-1">
|
||||||
{{template "article-banner-template" .}}
|
{{template "article-banner-template" .}}
|
||||||
|
|
||||||
@ -11,7 +11,7 @@
|
|||||||
<input name="article-title" type="text" value="{{.Article.Title}}" />
|
<input name="article-title" type="text" value="{{.Article.Title}}" />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="grid grid-cols-1 items-center">
|
<div class="flex">
|
||||||
<label class="btn text-center" for="article-banner">Titelbild</label>
|
<label class="btn text-center" for="article-banner">Titelbild</label>
|
||||||
<input class="hidden" id="article-banner" name="article-banner" type="file" required
|
<input class="hidden" id="article-banner" name="article-banner" type="file" required
|
||||||
hx-post="/article/upload-banner" hx-swap="outerHTML" hx-target="#article-banner-container" />
|
hx-post="/article/upload-banner" hx-swap="outerHTML" hx-target="#article-banner-container" />
|
||||||
@ -19,20 +19,14 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex flex-col gap-y-1">
|
<div class="flex flex-col gap-1">
|
||||||
<label for="article-summary">Beschreibung</label>
|
<label for="article-summary">Beschreibung</label>
|
||||||
<textarea name="article-summary">{{.Article.Summary}}</textarea>
|
<textarea name="article-summary">{{.Article.Summary}}</textarea>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex flex-col gap-y-1">
|
|
||||||
<label for="easyMDE">Artikel</label>
|
|
||||||
<textarea id="easyMDE">{{.Content}}</textarea>
|
|
||||||
<input id="article-content" name="article-content" type="hidden" value="{{.Content}}" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<span>Tags</span>
|
<span>Tags</span>
|
||||||
<div class="flex flex-wrap gap-x-4">
|
<div class="flex flex-wrap gap-4">
|
||||||
<div>
|
<div>
|
||||||
<input id="issue" name="issue" type="checkbox" {{if .Article.IsInIssue}}checked{{end}} />
|
<input id="issue" name="issue" type="checkbox" {{if .Article.IsInIssue}}checked{{end}} />
|
||||||
<label for="issue">Orient Express</label>
|
<label for="issue">Orient Express</label>
|
||||||
@ -48,6 +42,12 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="flex flex-col gap-1">
|
||||||
|
<label for="easyMDE">Artikel</label>
|
||||||
|
<textarea id="easyMDE">{{.Content}}</textarea>
|
||||||
|
<input id="article-content" name="article-content" type="hidden" value="{{.Content}}" />
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="btn-area">
|
<div class="btn-area">
|
||||||
<input class="action-btn" type="submit" value="Senden" hx-post="/article/{{.Action}}"
|
<input class="action-btn" type="submit" value="Senden" hx-post="/article/{{.Action}}"
|
||||||
hx-target="#page-content" />
|
hx-target="#page-content" />
|
||||||
@ -90,7 +90,7 @@
|
|||||||
|
|
||||||
{{define "article-banner-template"}}
|
{{define "article-banner-template"}}
|
||||||
<div id="article-banner-container">
|
<div id="article-banner-container">
|
||||||
<img src="/image/serve/{{.BannerImage}}" alt="">
|
<img src="/image/serve/{{.Image}}" alt="" />
|
||||||
<input id="article-banner-url" name="article-banner-url" type="hidden" value="{{.BannerImage}}" />
|
<input id="article-banner-url" name="article-banner-url" type="hidden" value="{{.Image}}" />
|
||||||
</div>
|
</div>
|
||||||
{{end}}
|
{{end}}
|
||||||
|
@ -1,46 +0,0 @@
|
|||||||
{{define "page-content"}}
|
|
||||||
<h2>Erster Benutzer (Administrator)</h2>
|
|
||||||
|
|
||||||
<form>
|
|
||||||
<div class="grid grid-cols-3 gap-4">
|
|
||||||
<div>
|
|
||||||
<label for="username">Benutzername</label>
|
|
||||||
<input class="w-full" required name="username" type="text" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password">Passwort</label>
|
|
||||||
<input class="w-full" required name="password" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="password2">Passwort wiederholen</label>
|
|
||||||
<input class="w-full" required name="password2" placeholder="***" type="password" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="first-name">Vorname</label>
|
|
||||||
<input class="w-full" required name="first-name" type="text" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="last-name">Nachname</label>
|
|
||||||
<input class="w-full" required name="last-name" type="text" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email">Emailadresse</label>
|
|
||||||
<input class="w-full" required name="email" type="text" />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label for="email2">Emailadresse wiederholen</label>
|
|
||||||
<input class="w-full" required name="email2" type="text" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="btn-area-1">
|
|
||||||
<input class="action-btn" type="submit" value="Anlegen" hx-post="/user/add-first" hx-target="#page-content" />
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
{{end}}
|
|
@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
<div>
|
<div>
|
||||||
<div class="w-full" id="article-banner-container">
|
<div class="w-full" id="article-banner-container">
|
||||||
<img src="/image/serve/{{.BannerImage}}" alt="Banner Image">
|
<img src="/image/serve/{{.Image}}" alt="Banner Image">
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<span>Titel</span>
|
<span>Titel</span>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user