Compare commits
2 Commits
23595e8008
...
e35da127aa
| Author | SHA1 | Date | |
|---|---|---|---|
| e35da127aa | |||
|
b93cdfba63
|
@@ -71,7 +71,11 @@ pub async fn page(Query(q): Query<LoginMsg>, req: Request) -> Result<Response, A
|
|||||||
// (if javascript is disabled, login via form still works)
|
// (if javascript is disabled, login via form still works)
|
||||||
script defer {(PreEscaped(r#"
|
script defer {(PreEscaped(r#"
|
||||||
if (window.location.search) {
|
if (window.location.search) {
|
||||||
history.replaceState(null, '', window.location.pathname);
|
const url = new URL(window.location.href);
|
||||||
|
const r = url.searchParams.get('r');
|
||||||
|
url.search = '';
|
||||||
|
if (r) url.searchParams.set('r', r);
|
||||||
|
history.replaceState(null, '', url.pathname + url.search);
|
||||||
}
|
}
|
||||||
document.getElementById('login-form').addEventListener('submit', async (e) => {
|
document.getElementById('login-form').addEventListener('submit', async (e) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@@ -89,7 +93,12 @@ pub async fn page(Query(q): Query<LoginMsg>, req: Request) -> Result<Response, A
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
window.location.href = '/dashboard';
|
const r = new URL(window.location.href).searchParams.get('r');
|
||||||
|
if (r && r.startsWith('/')) {
|
||||||
|
window.location.href = r;
|
||||||
|
} else {
|
||||||
|
window.location.href = '/dashboard';
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
const text = await res.text();
|
const text = await res.text();
|
||||||
err.textContent = text || 'Login failed';
|
err.textContent = text || 'Login failed';
|
||||||
|
|||||||
@@ -9,12 +9,14 @@ use crate::{
|
|||||||
error::CompositeError,
|
error::CompositeError,
|
||||||
logs::LogEntry,
|
logs::LogEntry,
|
||||||
users::{User, auth::UserAuthenticate, permissions::Permission},
|
users::{User, auth::UserAuthenticate, permissions::Permission},
|
||||||
web::{RedirectViaError, components::nav::nav, icons, pages::base},
|
web::{components::nav::nav, icons, pages::base},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, CompositeError> {
|
pub async fn page(req: Request) -> Result<Response, CompositeError> {
|
||||||
let u = User::authenticate(req.headers())?
|
let u = match User::authenticate(req.headers())? {
|
||||||
.ok_or(RedirectViaError(Redirect::to("/login?re=/logs")))?;
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let mut conn = database::conn()?;
|
let mut conn = database::conn()?;
|
||||||
let tx = conn.transaction()?;
|
let tx = conn.transaction()?;
|
||||||
let logs = LogEntry::get_all(&tx)?;
|
let logs = LogEntry::get_all(&tx)?;
|
||||||
@@ -63,7 +65,7 @@ pub async fn page(req: Request) -> Result<Response, CompositeError> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
p class="text-center p-2" {"You must have permission to view this page."}
|
p class="text-center p-2" {"You must have permission to view logs."}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -20,70 +20,69 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let mut conn = database::conn()?;
|
let mut conn = database::conn()?;
|
||||||
let tx = conn.transaction()?;
|
let tx = conn.transaction()?;
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"Persons | Mnemosyne",
|
"Persons | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
@if let Some(_) = u {
|
div class="mx-auto max-w-4xl px-2 my-4" {
|
||||||
div class="mx-auto max-w-4xl px-2 my-4" {
|
p class="flex items-center gap-2" {
|
||||||
p class="flex items-center gap-2" {
|
span class="text-neutral-500" {(PreEscaped(icons::CONTACT))}
|
||||||
span class="text-neutral-500" {(PreEscaped(icons::CONTACT))}
|
span class="text-2xl font-semibold font-lora" {"Persons"}
|
||||||
span class="text-2xl font-semibold font-lora" {"Persons"}
|
}
|
||||||
|
p class="text-neutral-500 text-sm font-light" {
|
||||||
|
@if let Ok(c) = Person::total_count(&tx) {
|
||||||
|
(c) " persons in total."
|
||||||
|
} @else {
|
||||||
|
"Could not get total person count."
|
||||||
}
|
}
|
||||||
p class="text-neutral-500 text-sm font-light" {
|
}
|
||||||
@if let Ok(c) = Person::total_count(&tx) {
|
}
|
||||||
(c) " persons in total."
|
@if let Ok(persons) = Person::get_all(&tx) {
|
||||||
} @else {
|
div class="max-w-4xl mx-auto px-2 mt-4 flex flex-wrap gap-2" {
|
||||||
"Could not get total person count."
|
@for person in &persons {
|
||||||
|
div class="rounded px-4 py-2 bg-neutral-200/10 border border-neutral-200/15 flex items-center" {
|
||||||
|
span class="text-neutral-400 mr-1 scale-125" {"~"}
|
||||||
|
span class="text-sm" {(person.primary_name)}
|
||||||
|
div class="w-px h-2/3 my-auto mx-2 bg-neutral-200/15" {}
|
||||||
|
div class="text-xs flex items-center" {
|
||||||
|
(
|
||||||
|
if let Ok(i) = person.get_in_quote_count(&tx) {
|
||||||
|
i.to_string()
|
||||||
|
} else {
|
||||||
|
"?".to_string()
|
||||||
|
}
|
||||||
|
) span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::SCROLL_TEXT))}
|
||||||
|
// div class="ml-2" {}
|
||||||
|
// "4" span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::FILE_IMAGE))}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@if let Ok(persons) = Person::get_all(&tx) {
|
@if persons.is_empty() {
|
||||||
div class="max-w-4xl mx-auto px-2 mt-4 flex flex-wrap gap-2" {
|
p class="text-center p-2" {"No persons yet."}
|
||||||
@for person in &persons {
|
}
|
||||||
div class="rounded px-4 py-2 bg-neutral-200/10 border border-neutral-200/15 flex items-center" {
|
div class="mx-auto max-w-4xl mt-4 px-2" {
|
||||||
span class="text-neutral-400 mr-1 scale-125" {"~"}
|
h3 class="font-lora font-semibold text-xl" {"Add new person"}
|
||||||
span class="text-sm" {(person.primary_name)}
|
form action="/persons/create" method="post" {
|
||||||
div class="w-px h-2/3 my-auto mx-2 bg-neutral-200/15" {}
|
label for="primary_name" class="text-neutral-500 font-light mt-2" {"Primary Name"}
|
||||||
div class="text-xs flex items-center" {
|
div class="flex gap-2" {
|
||||||
(
|
input type="text" autocomplete="off" id="primary_name" name="primary_name" placeholder="e.g. Frank"
|
||||||
if let Ok(i) = person.get_in_quote_count(&tx) {
|
class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
||||||
i.to_string()
|
button type="submit"
|
||||||
} else {
|
class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {"Submit"}
|
||||||
"?".to_string()
|
|
||||||
}
|
|
||||||
) span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::SCROLL_TEXT))}
|
|
||||||
// div class="ml-2" {}
|
|
||||||
// "4" span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::FILE_IMAGE))}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@if persons.is_empty() {
|
|
||||||
p class="text-center p-2" {"No persons yet."}
|
|
||||||
}
|
|
||||||
div class="mx-auto max-w-4xl mt-4 px-2" {
|
|
||||||
h3 class="font-lora font-semibold text-xl" {"Add new person"}
|
|
||||||
form action="/persons/create" method="post" {
|
|
||||||
label for="primary_name" class="text-neutral-500 font-light mt-2" {"Primary Name"}
|
|
||||||
div class="flex gap-2" {
|
|
||||||
input type="text" autocomplete="off" id="primary_name" name="primary_name" placeholder="e.g. Frank"
|
|
||||||
class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
|
||||||
button type="submit"
|
|
||||||
class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {"Submit"}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} @else {
|
|
||||||
p class="text-red-400 text-center" {"Failed to load persons."}
|
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
p class="text-center p-2" {"You must be logged in to view this page."}
|
p class="text-red-400 text-center" {"Failed to load persons."}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
use axum::{
|
use axum::{
|
||||||
extract::{Query, Request},
|
extract::{Query, Request},
|
||||||
response::{IntoResponse, Response},
|
response::{IntoResponse, Redirect, Response},
|
||||||
};
|
};
|
||||||
use maud::{PreEscaped, html};
|
use maud::{PreEscaped, html};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
@@ -9,10 +9,7 @@ use crate::{
|
|||||||
database,
|
database,
|
||||||
error::CompositeError,
|
error::CompositeError,
|
||||||
quotes::Quote,
|
quotes::Quote,
|
||||||
users::{
|
users::{User, auth::UserAuthenticate},
|
||||||
User,
|
|
||||||
auth::{UserAuthRequired, UserAuthenticate},
|
|
||||||
},
|
|
||||||
web::{
|
web::{
|
||||||
components::{nav::nav, quote::quote},
|
components::{nav::nav, quote::quote},
|
||||||
icons,
|
icons,
|
||||||
@@ -31,7 +28,10 @@ pub async fn page(
|
|||||||
Query(query): Query<PageQuery>,
|
Query(query): Query<PageQuery>,
|
||||||
req: Request,
|
req: Request,
|
||||||
) -> Result<Response, CompositeError> {
|
) -> Result<Response, CompositeError> {
|
||||||
let u = User::authenticate(req.headers())?.required()?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let conn = database::conn()?;
|
let conn = database::conn()?;
|
||||||
|
|
||||||
let page = query.page.unwrap_or(1).max(1);
|
let page = query.page.unwrap_or(1).max(1);
|
||||||
|
|||||||
@@ -27,14 +27,17 @@ const LINE_ADD_RM_SCRIPT: &str = include_str!("line-add-rm.js");
|
|||||||
const PREFILL_TIME_SCRIPT: &str = include_str!("prefill-time.js");
|
const PREFILL_TIME_SCRIPT: &str = include_str!("prefill-time.js");
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, CompositeError> {
|
pub async fn page(req: Request) -> Result<Response, CompositeError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let conn = database::conn()?;
|
let conn = database::conn()?;
|
||||||
let names = Name::get_all(&conn)?;
|
let names = Name::get_all(&conn)?;
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"Add Quote | Mnemosyne",
|
"Add Quote | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
div class="max-w-4xl mx-auto px-2" {
|
div class="max-w-4xl mx-auto px-2" {
|
||||||
div class="my-4 flex justify-between" {
|
div class="my-4 flex justify-between" {
|
||||||
|
|||||||
@@ -20,69 +20,68 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let conn = database::conn()?;
|
let conn = database::conn()?;
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"Tags | Mnemosyne",
|
"Tags | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
@if let Some(_) = u {
|
div class="mx-auto max-w-4xl px-2 my-4" {
|
||||||
div class="mx-auto max-w-4xl px-2 my-4" {
|
p class="flex items-center gap-2" {
|
||||||
p class="flex items-center gap-2" {
|
span class="text-neutral-500" {(PreEscaped(icons::TAG))}
|
||||||
span class="text-neutral-500" {(PreEscaped(icons::TAG))}
|
span class="text-2xl font-semibold font-lora" {"Tags"}
|
||||||
span class="text-2xl font-semibold font-lora" {"Tags"}
|
}
|
||||||
|
p class="text-neutral-500 text-sm font-light" {
|
||||||
|
@if let Ok(c) = Tag::total_count(&conn) {
|
||||||
|
(c) " tags in total."
|
||||||
|
} @else {
|
||||||
|
"Could not get total tag count."
|
||||||
}
|
}
|
||||||
p class="text-neutral-500 text-sm font-light" {
|
}
|
||||||
@if let Ok(c) = Tag::total_count(&conn) {
|
}
|
||||||
(c) " tags in total."
|
@if let Ok(tags) = Tag::get_all(&conn) {
|
||||||
} @else {
|
div class="max-w-4xl mx-auto px-2 mt-4 flex flex-wrap gap-2" {
|
||||||
"Could not get total tag count."
|
@for tag in &tags {
|
||||||
|
div class="rounded-full px-3 py-1 bg-neutral-200/10 border border-neutral-200/15 flex" {
|
||||||
|
span class="text-neutral-400 text-sm" {"#"}
|
||||||
|
span class="text-sm" {(tag.name)}
|
||||||
|
div class="w-px h-2/3 my-auto mx-2 bg-neutral-200/15" {}
|
||||||
|
div class="text-xs flex items-center" {
|
||||||
|
(
|
||||||
|
if let Ok(i) = tag.get_tagged_quotes_count(&conn) {
|
||||||
|
i.to_string()
|
||||||
|
} else {
|
||||||
|
"?".to_string()
|
||||||
|
}
|
||||||
|
) span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::SCROLL_TEXT))}
|
||||||
|
// div class="ml-2" {}
|
||||||
|
// "0" span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::FILE_IMAGE))}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@if let Ok(tags) = Tag::get_all(&conn) {
|
@if tags.is_empty() {
|
||||||
div class="max-w-4xl mx-auto px-2 mt-4 flex flex-wrap gap-2" {
|
p class="text-center p-2" {"No tags yet. How about making one?"}
|
||||||
@for tag in &tags {
|
}
|
||||||
div class="rounded-full px-3 py-1 bg-neutral-200/10 border border-neutral-200/15 flex" {
|
div class="mx-auto max-w-4xl mt-4 px-2" {
|
||||||
span class="text-neutral-400 text-sm" {"#"}
|
h3 class="font-lora font-semibold text-xl" {"Add new tag"}
|
||||||
span class="text-sm" {(tag.name)}
|
form action="/tags/create" method="post" {
|
||||||
div class="w-px h-2/3 my-auto mx-2 bg-neutral-200/15" {}
|
label for="tagname" class="text-neutral-500 font-light mt-2" {"Tag Name"}
|
||||||
div class="text-xs flex items-center" {
|
div class="flex gap-2" {
|
||||||
(
|
input type="text" autocomplete="off" id="tagname" name="tagname" placeholder="e.g. fashion"
|
||||||
if let Ok(i) = tag.get_tagged_quotes_count(&conn) {
|
class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
||||||
i.to_string()
|
button type="submit"
|
||||||
} else {
|
class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {"Submit"}
|
||||||
"?".to_string()
|
|
||||||
}
|
|
||||||
) span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::SCROLL_TEXT))}
|
|
||||||
// div class="ml-2" {}
|
|
||||||
// "0" span class="*:size-3 ml-1 text-neutral-400" {(PreEscaped(icons::FILE_IMAGE))}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@if tags.is_empty() {
|
|
||||||
p class="text-center p-2" {"No tags yet. How about making one?"}
|
|
||||||
}
|
|
||||||
div class="mx-auto max-w-4xl mt-4 px-2" {
|
|
||||||
h3 class="font-lora font-semibold text-xl" {"Add new tag"}
|
|
||||||
form action="/tags/create" method="post" {
|
|
||||||
label for="tagname" class="text-neutral-500 font-light mt-2" {"Tag Name"}
|
|
||||||
div class="flex gap-2" {
|
|
||||||
input type="text" autocomplete="off" id="tagname" name="tagname" placeholder="e.g. fashion"
|
|
||||||
class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
|
||||||
button type="submit"
|
|
||||||
class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {"Submit"}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} @else {
|
|
||||||
p class="text-red-400 text-center" {"Failed to load tags."}
|
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
p class="text-center p-2" {"You must be logged in to view this page."}
|
p class="text-red-400 text-center" {"Failed to load tags."}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
use axum::{
|
use axum::{
|
||||||
extract::Request,
|
extract::Request,
|
||||||
response::{IntoResponse, Response},
|
response::{IntoResponse, Redirect, Response},
|
||||||
};
|
};
|
||||||
use maud::{PreEscaped, html};
|
use maud::{PreEscaped, html};
|
||||||
|
|
||||||
@@ -22,49 +22,45 @@ pub mod create;
|
|||||||
pub mod profile;
|
pub mod profile;
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
let conn = database::conn()?;
|
Some(u) => u,
|
||||||
let us = match u.is_some() {
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
true => User::get_all(&conn),
|
|
||||||
false => Ok(vec![]),
|
|
||||||
};
|
};
|
||||||
|
let conn = database::conn()?;
|
||||||
|
let us = User::get_all(&conn);
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"Users | Mnemosyne",
|
"Users | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
@if let Some(u) = u {
|
div class="mx-auto max-w-4xl px-2 my-4" {
|
||||||
div class="mx-auto max-w-4xl px-2 my-4" {
|
p class="flex items-center gap-2" {
|
||||||
p class="flex items-center gap-2" {
|
span class="text-neutral-500" {(PreEscaped(icons::USERS))}
|
||||||
span class="text-neutral-500" {(PreEscaped(icons::USERS))}
|
span class="text-2xl font-semibold font-lora" {"Users"}
|
||||||
span class="text-2xl font-semibold font-lora" {"Users"}
|
|
||||||
}
|
|
||||||
p class="text-neutral-500 text-sm font-light" {
|
|
||||||
@if let Ok(v) = &us {
|
|
||||||
(v.len()) " users registered with Mnemosyne."
|
|
||||||
} @else {
|
|
||||||
"Could not fetch user count."
|
|
||||||
}
|
|
||||||
@if let Ok(true) = u.has_permission(&conn, Permission::ManuallyCreateUsers) {
|
|
||||||
" "
|
|
||||||
a href="/users/create" class="text-blue-500 hover:text-blue-400 hover:underline" {
|
|
||||||
"Create a new user"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
div class="mx-auto max-w-4xl flex flex-wrap gap-4" {
|
p class="text-neutral-500 text-sm font-light" {
|
||||||
@if let Ok(vec) = &us {
|
@if let Ok(v) = &us {
|
||||||
@for user in vec {
|
(v.len()) " users registered with Mnemosyne."
|
||||||
(user_miniprofile(user))
|
|
||||||
}
|
|
||||||
} @else {
|
} @else {
|
||||||
p class="text-center py-4 text-light text-red-500" {"Failed to load users."}
|
"Could not fetch user count."
|
||||||
|
}
|
||||||
|
@if let Ok(true) = u.has_permission(&conn, Permission::ManuallyCreateUsers) {
|
||||||
|
" "
|
||||||
|
a href="/users/create" class="text-blue-500 hover:text-blue-400 hover:underline" {
|
||||||
|
"Create a new user"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} @else {
|
}
|
||||||
p class="text-center p-2" {"You must be logged in to view this page."}
|
div class="mx-auto max-w-4xl flex flex-wrap gap-4" {
|
||||||
|
@if let Ok(vec) = &us {
|
||||||
|
@for user in vec {
|
||||||
|
(user_miniprofile(user))
|
||||||
|
}
|
||||||
|
} @else {
|
||||||
|
p class="text-center py-4 text-light text-red-500" {"Failed to load users."}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -21,42 +21,41 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
let conn = database::conn()?;
|
let conn = database::conn()?;
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"Create User | Mnemosyne",
|
"Create User | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
@if let Some(u) = u {
|
div class="mx-auto max-w-4xl px-2 my-4" {
|
||||||
div class="mx-auto max-w-4xl px-2 my-4" {
|
p class="flex items-center gap-2" {
|
||||||
p class="flex items-center gap-2" {
|
span class="text-neutral-500" {(PreEscaped(icons::USER_PLUS))}
|
||||||
span class="text-neutral-500" {(PreEscaped(icons::USER_PLUS))}
|
span class="text-2xl font-semibold font-lora" {"Create a new user"}
|
||||||
span class="text-2xl font-semibold font-lora" {"Create a new user"}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
@if let Ok(true) = u.has_permission(&conn, Permission::ManuallyCreateUsers) {
|
}
|
||||||
div class="mx-auto max-w-4xl px-2 mt-4" {
|
@if let Ok(true) = u.has_permission(&conn, Permission::ManuallyCreateUsers) {
|
||||||
form action="/users/create-form" method="post" class="flex flex-col" {
|
div class="mx-auto max-w-4xl px-2 mt-4" {
|
||||||
label for="handle" class="font-light text-neutral-500" {"Handle"}
|
form action="/users/create-form" method="post" class="flex flex-col" {
|
||||||
div class="flex w-64 items-center border border-neutral-200/25 rounded bg-neutral-950/50" {
|
label for="handle" class="font-light text-neutral-500" {"Handle"}
|
||||||
span class="pl-2 text-neutral-500 select-none" {"@"}
|
div class="flex w-64 items-center border border-neutral-200/25 rounded bg-neutral-950/50" {
|
||||||
input id="handle" name="handle" type="text" autocomplete="off"
|
span class="pl-2 text-neutral-500 select-none" {"@"}
|
||||||
class="w-fit pl-0.5 pr-1 py-1 outline-none";
|
input id="handle" name="handle" type="text" autocomplete="off"
|
||||||
}
|
class="w-fit pl-0.5 pr-1 py-1 outline-none";
|
||||||
label for="password" class="font-light text-neutral-500 mt-4" {"Password"} br;
|
|
||||||
input id="password" name="password" type="password" autocomplete="off"
|
|
||||||
class="px-2 w-64 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
|
||||||
input type="submit" value="Create"
|
|
||||||
class="px-4 mt-4 w-64 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40";
|
|
||||||
}
|
}
|
||||||
|
label for="password" class="font-light text-neutral-500 mt-4" {"Password"} br;
|
||||||
|
input id="password" name="password" type="password" autocomplete="off"
|
||||||
|
class="px-2 w-64 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
||||||
|
input type="submit" value="Create"
|
||||||
|
class="px-4 mt-4 w-64 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40";
|
||||||
}
|
}
|
||||||
} @else {
|
|
||||||
p class="text-center p-2" {"You must have permission to view this page."}
|
|
||||||
}
|
}
|
||||||
} @else {
|
} @else {
|
||||||
p class="text-center p-2" {"You must be logged in to view this page."}
|
p class="text-center p-2" {"You must have permission to view this page."}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ use crate::{
|
|||||||
pub async fn page(Path(id): Path<Uuid>, req: Request) -> Result<Response, CompositeError> {
|
pub async fn page(Path(id): Path<Uuid>, req: Request) -> Result<Response, CompositeError> {
|
||||||
let u = match User::authenticate(req.headers())? {
|
let u = match User::authenticate(req.headers())? {
|
||||||
Some(u) => u,
|
Some(u) => u,
|
||||||
None => return Ok(Redirect::to("/users").into_response()),
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
};
|
};
|
||||||
let mut conn = database::conn()?;
|
let mut conn = database::conn()?;
|
||||||
let tx = conn.transaction()?;
|
let tx = conn.transaction()?;
|
||||||
|
|||||||
@@ -20,53 +20,52 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
pub async fn page(req: Request) -> Result<Response, AuthError> {
|
||||||
let u = User::authenticate(req.headers())?;
|
let u = match User::authenticate(req.headers())? {
|
||||||
|
Some(u) => u,
|
||||||
|
None => return Ok(Redirect::to(&format!("/login?r={}", req.uri().path())).into_response()),
|
||||||
|
};
|
||||||
|
|
||||||
Ok(base(
|
Ok(base(
|
||||||
"User Settings | Mnemosyne",
|
"User Settings | Mnemosyne",
|
||||||
html!(
|
html!(
|
||||||
(nav(u.as_ref(), req.uri().path()))
|
(nav(Some(&u), req.uri().path()))
|
||||||
|
|
||||||
@if let Some(u) = u {
|
div class="max-w-4xl mx-auto px-2" {
|
||||||
div class="max-w-4xl mx-auto px-2" {
|
div class="mx-auto max-w-4xl my-4" {
|
||||||
div class="mx-auto max-w-4xl my-4" {
|
p class="flex items-center gap-2" {
|
||||||
p class="flex items-center gap-2" {
|
span class="text-neutral-500" {(PreEscaped(icons::SERVER))}
|
||||||
span class="text-neutral-500" {(PreEscaped(icons::SERVER))}
|
span class="text-2xl font-semibold font-lora" {"Your User Settings"}
|
||||||
span class="text-2xl font-semibold font-lora" {"Your User Settings"}
|
|
||||||
}
|
|
||||||
p class="text-neutral-500 text-sm font-light" {
|
|
||||||
// "Hi, " (u.handle) "!" " " "This is your user settings page." br;
|
|
||||||
"Looking for Mnemosyne settings?" " "
|
|
||||||
a class="text-blue-500 hover:text-blue-400 hover:underline" href="/mnemosyne-settings" {"Here."}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
p class="text-neutral-500 text-sm font-light" {
|
||||||
label for="handle" class="font-light text-neutral-500" {"Handle"}
|
// "Hi, " (u.handle) "!" " " "This is your user settings page." br;
|
||||||
form action="/user-settings/handle" method="post" class="flex gap-2" {
|
"Looking for Mnemosyne settings?" " "
|
||||||
div class="flex items-center border border-neutral-200/25 rounded bg-neutral-950/50" {
|
a class="text-blue-500 hover:text-blue-400 hover:underline" href="/mnemosyne-settings" {"Here."}
|
||||||
span class="pl-2 text-neutral-500 select-none" {"@"}
|
}
|
||||||
input id="handle" name="handle" type="text" autocomplete="off" value={(u.handle)}
|
}
|
||||||
class="w-full bg-transparent pl-0.5 pr-1 py-1 outline-none";
|
|
||||||
}
|
label for="handle" class="font-light text-neutral-500" {"Handle"}
|
||||||
button type="submit" class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {
|
form action="/user-settings/handle" method="post" class="flex gap-2" {
|
||||||
"Save"
|
div class="flex items-center border border-neutral-200/25 rounded bg-neutral-950/50" {
|
||||||
}
|
span class="pl-2 text-neutral-500 select-none" {"@"}
|
||||||
}
|
input id="handle" name="handle" type="text" autocomplete="off" value={(u.handle)}
|
||||||
hr class="mt-6 mb-4 border-neutral-600";
|
class="w-full bg-transparent pl-0.5 pr-1 py-1 outline-none";
|
||||||
p class="flex items-center gap-1" {
|
}
|
||||||
span class="text-neutral-500 scale-[.8]" {(PreEscaped(icons::USER_KEY))}
|
button type="submit" class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {
|
||||||
span class="text-lg font-semibold font-lora" {"Change Password"}
|
"Save"
|
||||||
}
|
}
|
||||||
label for="password" class="font-light text-neutral-500" {"New password"}
|
}
|
||||||
form action="/user-settings/passwd" method="post" class="flex gap-2" {
|
hr class="mt-6 mb-4 border-neutral-600";
|
||||||
input id="password" name="password" type="password" autocomplete="off" class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
p class="flex items-center gap-1" {
|
||||||
button type="submit" class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {
|
span class="text-neutral-500 scale-[.8]" {(PreEscaped(icons::USER_KEY))}
|
||||||
"Submit"
|
span class="text-lg font-semibold font-lora" {"Change Password"}
|
||||||
}
|
}
|
||||||
|
label for="password" class="font-light text-neutral-500" {"New password"}
|
||||||
|
form action="/user-settings/passwd" method="post" class="flex gap-2" {
|
||||||
|
input id="password" name="password" type="password" autocomplete="off" class="px-2 py-1 border border-neutral-200/25 bg-neutral-950/50 rounded";
|
||||||
|
button type="submit" class="px-4 py-1 border border-neutral-200/25 bg-neutral-200/5 rounded cursor-pointer hover:border-neutral-200/40" {
|
||||||
|
"Submit"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} @else {
|
|
||||||
p class="text-center p-2" {"You must be logged in to view this page."}
|
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user