mirror of
https://github.com/osmarks/mycorrhiza.git
synced 2025-01-22 16:16:51 +00:00
New user form in /admin/users/
This commit is contained in:
parent
d4fea3d24a
commit
1c24450a8f
@ -313,9 +313,6 @@ mark { background: rgba(130, 80, 30, 5); color: inherit; }
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* handlerug: sorry but I can't write in that unique and very special way */
|
|
||||||
/* i have to resort to the BORING way of writing CSS */
|
|
||||||
/* bouncepaw: they say that the best codes style is the consistent code style ☝️ */
|
|
||||||
kbd {
|
kbd {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
min-width: 1.5ch;
|
min-width: 1.5ch;
|
||||||
@ -382,7 +379,7 @@ kbd {
|
|||||||
.shortcuts-help .dialog__content {
|
.shortcuts-help .dialog__content {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
|
||||||
grid-column-gap: 32px;
|
grid-column-gap: 32px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.shortcuts-group-heading {
|
.shortcuts-group-heading {
|
||||||
@ -435,3 +432,55 @@ kbd {
|
|||||||
.table-cell--fill {
|
.table-cell--fill {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Form fields
|
||||||
|
*/
|
||||||
|
|
||||||
|
.form-field {
|
||||||
|
margin: 1em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-field label {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 600px) {
|
||||||
|
.form-field {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 150px max-content;
|
||||||
|
grid-column-gap: 16px;
|
||||||
|
}
|
||||||
|
.form-field label {
|
||||||
|
grid-column: 1;
|
||||||
|
}
|
||||||
|
.form-field input,
|
||||||
|
.form-field button,
|
||||||
|
.form-field select,
|
||||||
|
.form-field textarea,
|
||||||
|
.form-field .form-field__input {
|
||||||
|
grid-column: 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Notices
|
||||||
|
*/
|
||||||
|
|
||||||
|
.notice {
|
||||||
|
margin: 0.5em 0;
|
||||||
|
border: 1px solid;
|
||||||
|
padding: 0.5em 0.7em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.notice--error {
|
||||||
|
border-color: #e15757;
|
||||||
|
background-color: #ffb1b1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
.notice--error {
|
||||||
|
border-color: #a84343;
|
||||||
|
background-color: #5b3535;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -41,10 +41,12 @@ func Register(username, password, group string, force bool) error {
|
|||||||
switch {
|
switch {
|
||||||
case !util.IsPossibleUsername(username):
|
case !util.IsPossibleUsername(username):
|
||||||
return fmt.Errorf("illegal username \"%s\"", username)
|
return fmt.Errorf("illegal username \"%s\"", username)
|
||||||
|
case !ValidGroup(group):
|
||||||
|
return fmt.Errorf("invalid group \"%s\"", group)
|
||||||
|
case HasUsername(username):
|
||||||
|
return fmt.Errorf("username \"%s\" is already taken", username)
|
||||||
case !force && cfg.RegistrationLimit > 0 && Count() >= cfg.RegistrationLimit:
|
case !force && cfg.RegistrationLimit > 0 && Count() >= cfg.RegistrationLimit:
|
||||||
return fmt.Errorf("reached the limit of registered users (%d)", cfg.RegistrationLimit)
|
return fmt.Errorf("reached the limit of registered users (%d)", cfg.RegistrationLimit)
|
||||||
case !force && HasUsername(username):
|
|
||||||
return fmt.Errorf("username \"%s\" is already taken", username)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
hash, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
|
hash, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
|
||||||
|
46
util/util.go
46
util/util.go
@ -91,3 +91,49 @@ func HyphaNameFromRq(rq *http.Request, actions ...string) string {
|
|||||||
log.Println("HyphaNameFromRq: this request is invalid, fall back to home hypha")
|
log.Println("HyphaNameFromRq: this request is invalid, fall back to home hypha")
|
||||||
return cfg.HomeHypha
|
return cfg.HomeHypha
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FormData is a convenient struct for passing user input and errors to HTML
|
||||||
|
// forms and showing to the user.
|
||||||
|
type FormData struct {
|
||||||
|
err error
|
||||||
|
fields map[string]string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewFormData() FormData {
|
||||||
|
return FormData{
|
||||||
|
err: nil,
|
||||||
|
fields: map[string]string{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func FormDataFromRequest(r *http.Request, keys []string) FormData {
|
||||||
|
formData := NewFormData()
|
||||||
|
for _, key := range keys {
|
||||||
|
formData.Put(key, r.FormValue(key))
|
||||||
|
}
|
||||||
|
return formData
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FormData) HasError() bool {
|
||||||
|
return f.err != nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FormData) Error() string {
|
||||||
|
if f.err == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return f.err.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FormData) WithError(err error) FormData {
|
||||||
|
f.err = err
|
||||||
|
return f
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FormData) Get(key string) string {
|
||||||
|
return f.fields[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f FormData) Put(key, value string) {
|
||||||
|
f.fields[key] = value
|
||||||
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
{% import "github.com/bouncepaw/mycorrhiza/cfg" %}
|
{% import "github.com/bouncepaw/mycorrhiza/cfg" %}
|
||||||
{% import "github.com/bouncepaw/mycorrhiza/user" %}
|
{% import "github.com/bouncepaw/mycorrhiza/user" %}
|
||||||
|
{% import "github.com/bouncepaw/mycorrhiza/util" %}
|
||||||
|
|
||||||
{% func AdminPanelHTML() %}
|
{% func AdminPanelHTML() %}
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
@ -39,10 +40,11 @@
|
|||||||
<h1>Manage users</h1>
|
<h1>Manage users</h1>
|
||||||
|
|
||||||
<form action="/admin/reindex-users" method="post">
|
<form action="/admin/reindex-users" method="post">
|
||||||
|
<a class="btn" href="/admin/user/new">Create a new user</a>
|
||||||
<button class="btn" type="submit">Reindex users</button>
|
<button class="btn" type="submit">Reindex users</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<h2>Users list</h2>
|
<br>
|
||||||
|
|
||||||
<table class="users-table">
|
<table class="users-table">
|
||||||
<thead>
|
<thead>
|
||||||
@ -78,6 +80,51 @@
|
|||||||
</div>
|
</div>
|
||||||
{% endfunc %}
|
{% endfunc %}
|
||||||
|
|
||||||
|
{% func AdminUserNewHTML(formData util.FormData) %}
|
||||||
|
<div class="layout">
|
||||||
|
<main class="main-width">
|
||||||
|
<h1>New user</h1>
|
||||||
|
|
||||||
|
{% if formData.HasError() %}
|
||||||
|
<div class="notice notice--error">
|
||||||
|
<strong>Error:</strong>
|
||||||
|
{%s formData.Error() %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<form action="" method="post">
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="name">Name:</label>
|
||||||
|
<input type="text" name="name" id="name" value="{%s formData.Get("name") %}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="password">Password:</label>
|
||||||
|
<input type="password" name="password" id="password" value="{%s formData.Get("password") %}">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="group">Group:</label>
|
||||||
|
<select id="group" name="group">
|
||||||
|
<option{% if formData.Get("group") == "anon" %} selected{% endif %}>anon</option>
|
||||||
|
<option{% if formData.Get("group") == "editor" %} selected{% endif %}>editor</option>
|
||||||
|
<option{% if formData.Get("group") == "trusted" %} selected{% endif %}>trusted</option>
|
||||||
|
<option{% if formData.Get("group") == "moderator" %} selected{% endif %}>moderator</option>
|
||||||
|
<option{% if formData.Get("group") == "admin" %} selected{% endif %}>admin</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<div class="form-field__input">
|
||||||
|
<button class="btn" type="submit">Create</button>
|
||||||
|
<a class="btn btn_weak" href="/admin/users/">Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
{% endfunc %}
|
||||||
|
|
||||||
{% func AdminUsersUserHTML(u *user.User) %}
|
{% func AdminUsersUserHTML(u *user.User) %}
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
<main class="main-width">
|
<main class="main-width">
|
||||||
|
@ -10,22 +10,25 @@ import "github.com/bouncepaw/mycorrhiza/cfg"
|
|||||||
//line views/admin.qtpl:2
|
//line views/admin.qtpl:2
|
||||||
import "github.com/bouncepaw/mycorrhiza/user"
|
import "github.com/bouncepaw/mycorrhiza/user"
|
||||||
|
|
||||||
//line views/admin.qtpl:4
|
//line views/admin.qtpl:3
|
||||||
|
import "github.com/bouncepaw/mycorrhiza/util"
|
||||||
|
|
||||||
|
//line views/admin.qtpl:5
|
||||||
import (
|
import (
|
||||||
qtio422016 "io"
|
qtio422016 "io"
|
||||||
|
|
||||||
qt422016 "github.com/valyala/quicktemplate"
|
qt422016 "github.com/valyala/quicktemplate"
|
||||||
)
|
)
|
||||||
|
|
||||||
//line views/admin.qtpl:4
|
//line views/admin.qtpl:5
|
||||||
var (
|
var (
|
||||||
_ = qtio422016.Copy
|
_ = qtio422016.Copy
|
||||||
_ = qt422016.AcquireByteBuffer
|
_ = qt422016.AcquireByteBuffer
|
||||||
)
|
)
|
||||||
|
|
||||||
//line views/admin.qtpl:4
|
//line views/admin.qtpl:5
|
||||||
func StreamAdminPanelHTML(qw422016 *qt422016.Writer) {
|
func StreamAdminPanelHTML(qw422016 *qt422016.Writer) {
|
||||||
//line views/admin.qtpl:4
|
//line views/admin.qtpl:5
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
<main class="main-width">
|
<main class="main-width">
|
||||||
@ -57,48 +60,49 @@ func StreamAdminPanelHTML(qw422016 *qt422016.Writer) {
|
|||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
func WriteAdminPanelHTML(qq422016 qtio422016.Writer) {
|
func WriteAdminPanelHTML(qq422016 qtio422016.Writer) {
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
qw422016 := qt422016.AcquireWriter(qq422016)
|
qw422016 := qt422016.AcquireWriter(qq422016)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
StreamAdminPanelHTML(qw422016)
|
StreamAdminPanelHTML(qw422016)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
qt422016.ReleaseWriter(qw422016)
|
qt422016.ReleaseWriter(qw422016)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
func AdminPanelHTML() string {
|
func AdminPanelHTML() string {
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
qb422016 := qt422016.AcquireByteBuffer()
|
qb422016 := qt422016.AcquireByteBuffer()
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
WriteAdminPanelHTML(qb422016)
|
WriteAdminPanelHTML(qb422016)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
qs422016 := string(qb422016.B)
|
qs422016 := string(qb422016.B)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
qt422016.ReleaseByteBuffer(qb422016)
|
qt422016.ReleaseByteBuffer(qb422016)
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
return qs422016
|
return qs422016
|
||||||
//line views/admin.qtpl:34
|
//line views/admin.qtpl:35
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:36
|
//line views/admin.qtpl:37
|
||||||
func StreamAdminUsersPanelHTML(qw422016 *qt422016.Writer, userList []*user.User) {
|
func StreamAdminUsersPanelHTML(qw422016 *qt422016.Writer, userList []*user.User) {
|
||||||
//line views/admin.qtpl:36
|
//line views/admin.qtpl:37
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
<main class="main-width">
|
<main class="main-width">
|
||||||
<h1>Manage users</h1>
|
<h1>Manage users</h1>
|
||||||
|
|
||||||
<form action="/admin/reindex-users" method="post">
|
<form action="/admin/reindex-users" method="post">
|
||||||
|
<a class="btn" href="/admin/user/new">Create a new user</a>
|
||||||
<button class="btn" type="submit">Reindex users</button>
|
<button class="btn" type="submit">Reindex users</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
<h2>Users list</h2>
|
<br>
|
||||||
|
|
||||||
<table class="users-table">
|
<table class="users-table">
|
||||||
<thead>
|
<thead>
|
||||||
@ -111,111 +115,247 @@ func StreamAdminUsersPanelHTML(qw422016 *qt422016.Writer, userList []*user.User)
|
|||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:57
|
//line views/admin.qtpl:59
|
||||||
for _, u := range userList {
|
for _, u := range userList {
|
||||||
//line views/admin.qtpl:57
|
//line views/admin.qtpl:59
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
<tr>
|
<tr>
|
||||||
<td class="table-cell--fill">
|
<td class="table-cell--fill">
|
||||||
<a href="/hypha/`)
|
<a href="/hypha/`)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.N().U(cfg.UserHypha)
|
qw422016.N().U(cfg.UserHypha)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.N().S(`/`)
|
qw422016.N().S(`/`)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.N().U(u.Name)
|
qw422016.N().U(u.Name)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.N().S(`">`)
|
qw422016.N().S(`">`)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.E().S(u.Name)
|
qw422016.E().S(u.Name)
|
||||||
//line views/admin.qtpl:60
|
//line views/admin.qtpl:62
|
||||||
qw422016.N().S(`</a>
|
qw422016.N().S(`</a>
|
||||||
</td>
|
</td>
|
||||||
<td>`)
|
<td>`)
|
||||||
//line views/admin.qtpl:62
|
//line views/admin.qtpl:64
|
||||||
qw422016.E().S(u.Group)
|
qw422016.E().S(u.Group)
|
||||||
//line views/admin.qtpl:62
|
//line views/admin.qtpl:64
|
||||||
qw422016.N().S(`</td>
|
qw422016.N().S(`</td>
|
||||||
<td>
|
<td>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:64
|
//line views/admin.qtpl:66
|
||||||
if u.RegisteredAt.IsZero() {
|
if u.RegisteredAt.IsZero() {
|
||||||
//line views/admin.qtpl:64
|
//line views/admin.qtpl:66
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
unknown
|
unknown
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:66
|
//line views/admin.qtpl:68
|
||||||
} else {
|
} else {
|
||||||
//line views/admin.qtpl:66
|
//line views/admin.qtpl:68
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:67
|
//line views/admin.qtpl:69
|
||||||
qw422016.E().S(u.RegisteredAt.UTC().Format("2006-01-02 15:04"))
|
qw422016.E().S(u.RegisteredAt.UTC().Format("2006-01-02 15:04"))
|
||||||
//line views/admin.qtpl:67
|
//line views/admin.qtpl:69
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:68
|
//line views/admin.qtpl:70
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:68
|
//line views/admin.qtpl:70
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<a href="/admin/users/`)
|
<a href="/admin/users/`)
|
||||||
//line views/admin.qtpl:71
|
//line views/admin.qtpl:73
|
||||||
qw422016.N().U(u.Name)
|
qw422016.N().U(u.Name)
|
||||||
//line views/admin.qtpl:71
|
//line views/admin.qtpl:73
|
||||||
qw422016.N().S(`/edit">Edit</a>
|
qw422016.N().S(`/edit">Edit</a>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:74
|
//line views/admin.qtpl:76
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:74
|
//line views/admin.qtpl:76
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
func WriteAdminUsersPanelHTML(qq422016 qtio422016.Writer, userList []*user.User) {
|
func WriteAdminUsersPanelHTML(qq422016 qtio422016.Writer, userList []*user.User) {
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
qw422016 := qt422016.AcquireWriter(qq422016)
|
qw422016 := qt422016.AcquireWriter(qq422016)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
StreamAdminUsersPanelHTML(qw422016, userList)
|
StreamAdminUsersPanelHTML(qw422016, userList)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
qt422016.ReleaseWriter(qw422016)
|
qt422016.ReleaseWriter(qw422016)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
func AdminUsersPanelHTML(userList []*user.User) string {
|
func AdminUsersPanelHTML(userList []*user.User) string {
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
qb422016 := qt422016.AcquireByteBuffer()
|
qb422016 := qt422016.AcquireByteBuffer()
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
WriteAdminUsersPanelHTML(qb422016, userList)
|
WriteAdminUsersPanelHTML(qb422016, userList)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
qs422016 := string(qb422016.B)
|
qs422016 := string(qb422016.B)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
qt422016.ReleaseByteBuffer(qb422016)
|
qt422016.ReleaseByteBuffer(qb422016)
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
return qs422016
|
return qs422016
|
||||||
//line views/admin.qtpl:79
|
//line views/admin.qtpl:81
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:81
|
//line views/admin.qtpl:83
|
||||||
|
func StreamAdminUserNewHTML(qw422016 *qt422016.Writer, formData util.FormData) {
|
||||||
|
//line views/admin.qtpl:83
|
||||||
|
qw422016.N().S(`
|
||||||
|
<div class="layout">
|
||||||
|
<main class="main-width">
|
||||||
|
<h1>New user</h1>
|
||||||
|
|
||||||
|
`)
|
||||||
|
//line views/admin.qtpl:88
|
||||||
|
if formData.HasError() {
|
||||||
|
//line views/admin.qtpl:88
|
||||||
|
qw422016.N().S(`
|
||||||
|
<div class="notice notice--error">
|
||||||
|
<strong>Error:</strong>
|
||||||
|
`)
|
||||||
|
//line views/admin.qtpl:91
|
||||||
|
qw422016.E().S(formData.Error())
|
||||||
|
//line views/admin.qtpl:91
|
||||||
|
qw422016.N().S(`
|
||||||
|
</div>
|
||||||
|
`)
|
||||||
|
//line views/admin.qtpl:93
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:93
|
||||||
|
qw422016.N().S(`
|
||||||
|
|
||||||
|
<form action="" method="post">
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="name">Name:</label>
|
||||||
|
<input type="text" name="name" id="name" value="`)
|
||||||
|
//line views/admin.qtpl:98
|
||||||
|
qw422016.E().S(formData.Get("name"))
|
||||||
|
//line views/admin.qtpl:98
|
||||||
|
qw422016.N().S(`">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="password">Password:</label>
|
||||||
|
<input type="password" name="password" id="password" value="`)
|
||||||
|
//line views/admin.qtpl:103
|
||||||
|
qw422016.E().S(formData.Get("password"))
|
||||||
|
//line views/admin.qtpl:103
|
||||||
|
qw422016.N().S(`">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<label for="group">Group:</label>
|
||||||
|
<select id="group" name="group">
|
||||||
|
<option`)
|
||||||
|
//line views/admin.qtpl:109
|
||||||
|
if formData.Get("group") == "anon" {
|
||||||
|
//line views/admin.qtpl:109
|
||||||
|
qw422016.N().S(` selected`)
|
||||||
|
//line views/admin.qtpl:109
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:109
|
||||||
|
qw422016.N().S(`>anon</option>
|
||||||
|
<option`)
|
||||||
|
//line views/admin.qtpl:110
|
||||||
|
if formData.Get("group") == "editor" {
|
||||||
|
//line views/admin.qtpl:110
|
||||||
|
qw422016.N().S(` selected`)
|
||||||
|
//line views/admin.qtpl:110
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:110
|
||||||
|
qw422016.N().S(`>editor</option>
|
||||||
|
<option`)
|
||||||
|
//line views/admin.qtpl:111
|
||||||
|
if formData.Get("group") == "trusted" {
|
||||||
|
//line views/admin.qtpl:111
|
||||||
|
qw422016.N().S(` selected`)
|
||||||
|
//line views/admin.qtpl:111
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:111
|
||||||
|
qw422016.N().S(`>trusted</option>
|
||||||
|
<option`)
|
||||||
|
//line views/admin.qtpl:112
|
||||||
|
if formData.Get("group") == "moderator" {
|
||||||
|
//line views/admin.qtpl:112
|
||||||
|
qw422016.N().S(` selected`)
|
||||||
|
//line views/admin.qtpl:112
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:112
|
||||||
|
qw422016.N().S(`>moderator</option>
|
||||||
|
<option`)
|
||||||
|
//line views/admin.qtpl:113
|
||||||
|
if formData.Get("group") == "admin" {
|
||||||
|
//line views/admin.qtpl:113
|
||||||
|
qw422016.N().S(` selected`)
|
||||||
|
//line views/admin.qtpl:113
|
||||||
|
}
|
||||||
|
//line views/admin.qtpl:113
|
||||||
|
qw422016.N().S(`>admin</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-field">
|
||||||
|
<div class="form-field__input">
|
||||||
|
<button class="btn" type="submit">Create</button>
|
||||||
|
<a class="btn btn_weak" href="/admin/users/">Cancel</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
`)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
}
|
||||||
|
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
func WriteAdminUserNewHTML(qq422016 qtio422016.Writer, formData util.FormData) {
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
qw422016 := qt422016.AcquireWriter(qq422016)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
StreamAdminUserNewHTML(qw422016, formData)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
qt422016.ReleaseWriter(qw422016)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
}
|
||||||
|
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
func AdminUserNewHTML(formData util.FormData) string {
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
qb422016 := qt422016.AcquireByteBuffer()
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
WriteAdminUserNewHTML(qb422016, formData)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
qs422016 := string(qb422016.B)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
qt422016.ReleaseByteBuffer(qb422016)
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
return qs422016
|
||||||
|
//line views/admin.qtpl:126
|
||||||
|
}
|
||||||
|
|
||||||
|
//line views/admin.qtpl:128
|
||||||
func StreamAdminUsersUserHTML(qw422016 *qt422016.Writer, u *user.User) {
|
func StreamAdminUsersUserHTML(qw422016 *qt422016.Writer, u *user.User) {
|
||||||
//line views/admin.qtpl:81
|
//line views/admin.qtpl:128
|
||||||
qw422016.N().S(`
|
qw422016.N().S(`
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
<main class="main-width">
|
<main class="main-width">
|
||||||
<h1>`)
|
<h1>`)
|
||||||
//line views/admin.qtpl:84
|
//line views/admin.qtpl:131
|
||||||
qw422016.E().S(u.Name)
|
qw422016.E().S(u.Name)
|
||||||
//line views/admin.qtpl:84
|
//line views/admin.qtpl:131
|
||||||
qw422016.N().S(`</h1>
|
qw422016.N().S(`</h1>
|
||||||
|
|
||||||
<form action="" method="post">
|
<form action="" method="post">
|
||||||
@ -223,49 +363,49 @@ func StreamAdminUsersUserHTML(qw422016 *qt422016.Writer, u *user.User) {
|
|||||||
<label for="group">Group:</label>
|
<label for="group">Group:</label>
|
||||||
<select id="group" name="group">
|
<select id="group" name="group">
|
||||||
<option`)
|
<option`)
|
||||||
//line views/admin.qtpl:90
|
//line views/admin.qtpl:137
|
||||||
if u.Group == "anon" {
|
if u.Group == "anon" {
|
||||||
//line views/admin.qtpl:90
|
//line views/admin.qtpl:137
|
||||||
qw422016.N().S(` selected`)
|
qw422016.N().S(` selected`)
|
||||||
//line views/admin.qtpl:90
|
//line views/admin.qtpl:137
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:90
|
//line views/admin.qtpl:137
|
||||||
qw422016.N().S(`>anon</option>
|
qw422016.N().S(`>anon</option>
|
||||||
<option`)
|
<option`)
|
||||||
//line views/admin.qtpl:91
|
//line views/admin.qtpl:138
|
||||||
if u.Group == "editor" {
|
if u.Group == "editor" {
|
||||||
//line views/admin.qtpl:91
|
//line views/admin.qtpl:138
|
||||||
qw422016.N().S(` selected`)
|
qw422016.N().S(` selected`)
|
||||||
//line views/admin.qtpl:91
|
//line views/admin.qtpl:138
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:91
|
//line views/admin.qtpl:138
|
||||||
qw422016.N().S(`>editor</option>
|
qw422016.N().S(`>editor</option>
|
||||||
<option`)
|
<option`)
|
||||||
//line views/admin.qtpl:92
|
//line views/admin.qtpl:139
|
||||||
if u.Group == "trusted" {
|
if u.Group == "trusted" {
|
||||||
//line views/admin.qtpl:92
|
//line views/admin.qtpl:139
|
||||||
qw422016.N().S(` selected`)
|
qw422016.N().S(` selected`)
|
||||||
//line views/admin.qtpl:92
|
//line views/admin.qtpl:139
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:92
|
//line views/admin.qtpl:139
|
||||||
qw422016.N().S(`>trusted</option>
|
qw422016.N().S(`>trusted</option>
|
||||||
<option`)
|
<option`)
|
||||||
//line views/admin.qtpl:93
|
//line views/admin.qtpl:140
|
||||||
if u.Group == "moderator" {
|
if u.Group == "moderator" {
|
||||||
//line views/admin.qtpl:93
|
//line views/admin.qtpl:140
|
||||||
qw422016.N().S(` selected`)
|
qw422016.N().S(` selected`)
|
||||||
//line views/admin.qtpl:93
|
//line views/admin.qtpl:140
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:93
|
//line views/admin.qtpl:140
|
||||||
qw422016.N().S(`>moderator</option>
|
qw422016.N().S(`>moderator</option>
|
||||||
<option`)
|
<option`)
|
||||||
//line views/admin.qtpl:94
|
//line views/admin.qtpl:141
|
||||||
if u.Group == "admin" {
|
if u.Group == "admin" {
|
||||||
//line views/admin.qtpl:94
|
//line views/admin.qtpl:141
|
||||||
qw422016.N().S(` selected`)
|
qw422016.N().S(` selected`)
|
||||||
//line views/admin.qtpl:94
|
//line views/admin.qtpl:141
|
||||||
}
|
}
|
||||||
//line views/admin.qtpl:94
|
//line views/admin.qtpl:141
|
||||||
qw422016.N().S(`>admin</option>
|
qw422016.N().S(`>admin</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
@ -276,31 +416,31 @@ func StreamAdminUsersUserHTML(qw422016 *qt422016.Writer, u *user.User) {
|
|||||||
</main>
|
</main>
|
||||||
</div>
|
</div>
|
||||||
`)
|
`)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
func WriteAdminUsersUserHTML(qq422016 qtio422016.Writer, u *user.User) {
|
func WriteAdminUsersUserHTML(qq422016 qtio422016.Writer, u *user.User) {
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
qw422016 := qt422016.AcquireWriter(qq422016)
|
qw422016 := qt422016.AcquireWriter(qq422016)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
StreamAdminUsersUserHTML(qw422016, u)
|
StreamAdminUsersUserHTML(qw422016, u)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
qt422016.ReleaseWriter(qw422016)
|
qt422016.ReleaseWriter(qw422016)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
}
|
}
|
||||||
|
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
func AdminUsersUserHTML(u *user.User) string {
|
func AdminUsersUserHTML(u *user.User) string {
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
qb422016 := qt422016.AcquireByteBuffer()
|
qb422016 := qt422016.AcquireByteBuffer()
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
WriteAdminUsersUserHTML(qb422016, u)
|
WriteAdminUsersUserHTML(qb422016, u)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
qs422016 := string(qb422016.B)
|
qs422016 := string(qb422016.B)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
qt422016.ReleaseByteBuffer(qb422016)
|
qt422016.ReleaseByteBuffer(qb422016)
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
return qs422016
|
return qs422016
|
||||||
//line views/admin.qtpl:103
|
//line views/admin.qtpl:150
|
||||||
}
|
}
|
||||||
|
39
web/admin.go
39
web/admin.go
@ -23,6 +23,7 @@ func initAdmin() {
|
|||||||
http.HandleFunc("/admin/reindex-users/", handlerAdminReindexUsers)
|
http.HandleFunc("/admin/reindex-users/", handlerAdminReindexUsers)
|
||||||
|
|
||||||
http.HandleFunc("/admin/users/", handlerAdminUsers)
|
http.HandleFunc("/admin/users/", handlerAdminUsers)
|
||||||
|
http.HandleFunc("/admin/user/new", handlerAdminUserNew)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -124,7 +125,41 @@ func handlerAdminUsers(w http.ResponseWriter, r *http.Request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
util.HTTP404Page(w, "404 page not found")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
util.HTTP404Page(w, "404 page not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
func handlerAdminUserNew(w http.ResponseWriter, r *http.Request) {
|
||||||
|
util.PrepareRq(r)
|
||||||
|
if user.CanProceed(r, "admin") {
|
||||||
|
if r.Method == http.MethodGet {
|
||||||
|
// New user form
|
||||||
|
html := views.AdminUserNewHTML(util.NewFormData())
|
||||||
|
html = views.BaseHTML("New user", html, user.FromRequest(r))
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
||||||
|
io.WriteString(w, html)
|
||||||
|
return
|
||||||
|
} else if r.Method == http.MethodPost {
|
||||||
|
// Create a user
|
||||||
|
f := util.FormDataFromRequest(r, []string{"name", "password", "group"})
|
||||||
|
|
||||||
|
err := user.Register(f.Get("name"), f.Get("password"), f.Get("group"), true)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
html := views.AdminUserNewHTML(f.WithError(err))
|
||||||
|
html = views.BaseHTML("New user", html, user.FromRequest(r))
|
||||||
|
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
w.Header().Set("Content-Type", mime.TypeByExtension(".html"))
|
||||||
|
io.WriteString(w, html)
|
||||||
|
} else {
|
||||||
|
http.Redirect(w, r, "/admin/users/", http.StatusSeeOther)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
util.HTTP404Page(w, "404 page not found")
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user