mirror of
https://github.com/mCaptcha/mCaptcha.git
synced 2026-02-11 10:05:41 +00:00
notification date formatting
This commit is contained in:
@@ -51,6 +51,7 @@ impl From<Notification> for NotificationResp {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// route handler that gets all unread notifications
|
/// route handler that gets all unread notifications
|
||||||
#[my_codegen::get(
|
#[my_codegen::get(
|
||||||
path = "crate::V1_API_ROUTES.notifications.get",
|
path = "crate::V1_API_ROUTES.notifications.get",
|
||||||
@@ -63,7 +64,14 @@ pub async fn get_notification(
|
|||||||
let receiver = id.identity().unwrap();
|
let receiver = id.identity().unwrap();
|
||||||
// TODO handle error where payload.to doesnt exist
|
// TODO handle error where payload.to doesnt exist
|
||||||
|
|
||||||
let resp = runner::get_notification(&data, &receiver).await?;
|
let mut notifications = runner::get_notification(&data, &receiver).await?;
|
||||||
|
let resp: Vec<NotificationResp> = notifications
|
||||||
|
.drain(0..)
|
||||||
|
.map(|x| {
|
||||||
|
let y: NotificationResp = x.into();
|
||||||
|
y
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
Ok(HttpResponse::Ok().json(resp))
|
Ok(HttpResponse::Ok().json(resp))
|
||||||
}
|
}
|
||||||
@@ -73,10 +81,10 @@ pub mod runner {
|
|||||||
pub async fn get_notification(
|
pub async fn get_notification(
|
||||||
data: &AppData,
|
data: &AppData,
|
||||||
receiver: &str,
|
receiver: &str,
|
||||||
) -> ServiceResult<Vec<NotificationResp>> {
|
) -> ServiceResult<Vec<Notification>> {
|
||||||
// TODO handle error where payload.to doesnt exist
|
// TODO handle error where payload.to doesnt exist
|
||||||
|
|
||||||
let mut notifications = sqlx::query_file_as!(
|
let notifications = sqlx::query_file_as!(
|
||||||
Notification,
|
Notification,
|
||||||
"src/api/v1/notifications/get_all_unread.sql",
|
"src/api/v1/notifications/get_all_unread.sql",
|
||||||
&receiver
|
&receiver
|
||||||
@@ -84,15 +92,7 @@ pub mod runner {
|
|||||||
.fetch_all(&data.db)
|
.fetch_all(&data.db)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
let resp = notifications
|
Ok(notifications)
|
||||||
.drain(0..)
|
|
||||||
.map(|x| {
|
|
||||||
let y: NotificationResp = x.into();
|
|
||||||
y
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
Ok(resp)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -18,24 +18,72 @@
|
|||||||
use actix_identity::Identity;
|
use actix_identity::Identity;
|
||||||
use actix_web::{HttpResponse, Responder};
|
use actix_web::{HttpResponse, Responder};
|
||||||
use sailfish::TemplateOnce;
|
use sailfish::TemplateOnce;
|
||||||
|
use sqlx::types::time::OffsetDateTime;
|
||||||
|
|
||||||
use crate::api::v1::notifications::get::{runner, NotificationResp};
|
use crate::api::v1::notifications::get::{self, runner};
|
||||||
use crate::errors::PageResult;
|
use crate::errors::PageResult;
|
||||||
use crate::AppData;
|
use crate::AppData;
|
||||||
|
|
||||||
|
const MINUTE: i64 = 60;
|
||||||
|
const HOUR: i64 = MINUTE * 60;
|
||||||
|
const DAY: i64 = HOUR * 24;
|
||||||
|
const WEEK: i64 = DAY * 7;
|
||||||
|
|
||||||
#[derive(TemplateOnce)]
|
#[derive(TemplateOnce)]
|
||||||
#[template(path = "panel/notifications/index.html")]
|
#[template(path = "panel/notifications/index.html")]
|
||||||
pub struct IndexPage {
|
pub struct IndexPage {
|
||||||
/// notifications
|
/// notifications
|
||||||
n: Vec<NotificationResp>,
|
n: Vec<Notification>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl IndexPage {
|
impl IndexPage {
|
||||||
fn new(n: Vec<NotificationResp>) -> Self {
|
fn new(n: Vec<Notification>) -> Self {
|
||||||
IndexPage { n }
|
IndexPage { n }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub struct Notification {
|
||||||
|
pub name: String,
|
||||||
|
pub heading: String,
|
||||||
|
pub message: String,
|
||||||
|
pub received: OffsetDateTime,
|
||||||
|
pub id: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<get::Notification> for Notification {
|
||||||
|
fn from(n: get::Notification) -> Self {
|
||||||
|
Notification {
|
||||||
|
name: n.name.unwrap(),
|
||||||
|
heading: n.heading.unwrap(),
|
||||||
|
received: n.received.unwrap(),
|
||||||
|
id: n.id.unwrap(),
|
||||||
|
message: n.message.unwrap(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Notification {
|
||||||
|
pub fn print_date(&self) -> String {
|
||||||
|
let date = self.received;
|
||||||
|
let timestamp = self.received.unix_timestamp();
|
||||||
|
let now = OffsetDateTime::now_utc().unix_timestamp();
|
||||||
|
|
||||||
|
let difference = now - timestamp;
|
||||||
|
|
||||||
|
if difference >= 3 * WEEK {
|
||||||
|
date.format("%d-%m-%y")
|
||||||
|
} else if (DAY..(3 * WEEK)).contains(&difference) {
|
||||||
|
format!("{} days ago", date.hour())
|
||||||
|
} else if (HOUR..DAY).contains(&difference) {
|
||||||
|
format!("{} hours ago", date.hour())
|
||||||
|
} else if (MINUTE..HOUR).contains(&difference) {
|
||||||
|
format!("{} minutes ago", date.minute())
|
||||||
|
} else {
|
||||||
|
format!("{} seconds ago", date.second())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const PAGE: &str = "Notifications";
|
const PAGE: &str = "Notifications";
|
||||||
|
|
||||||
#[my_codegen::get(path = "crate::PAGES.panel.notifications", wrap = "crate::CheckLogin")]
|
#[my_codegen::get(path = "crate::PAGES.panel.notifications", wrap = "crate::CheckLogin")]
|
||||||
@@ -43,10 +91,56 @@ pub async fn notifications(data: AppData, id: Identity) -> PageResult<impl Respo
|
|||||||
let receiver = id.identity().unwrap();
|
let receiver = id.identity().unwrap();
|
||||||
// TODO handle error where payload.to doesnt exist
|
// TODO handle error where payload.to doesnt exist
|
||||||
|
|
||||||
let notifications = runner::get_notification(&data, &receiver).await?;
|
let mut notifications = runner::get_notification(&data, &receiver).await?;
|
||||||
|
let notifications = notifications.drain(0..).map(|x| x.into()).collect();
|
||||||
|
|
||||||
let body = IndexPage::new(notifications).render_once().unwrap();
|
let body = IndexPage::new(notifications).render_once().unwrap();
|
||||||
Ok(HttpResponse::Ok()
|
Ok(HttpResponse::Ok()
|
||||||
.content_type("text/html; charset=utf-8")
|
.content_type("text/html; charset=utf-8")
|
||||||
.body(body))
|
.body(body))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn print_date_test() {
|
||||||
|
let mut n = Notification {
|
||||||
|
received: OffsetDateTime::now_utc(),
|
||||||
|
name: String::default(),
|
||||||
|
heading: String::default(),
|
||||||
|
message: String::default(),
|
||||||
|
id: 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
let timestamp = n.received.unix_timestamp();
|
||||||
|
println!("timestamp: {}", timestamp);
|
||||||
|
|
||||||
|
// seconds test
|
||||||
|
assert!(n.print_date().contains("seconds ago"));
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - 5);
|
||||||
|
assert!(n.print_date().contains("seconds ago"));
|
||||||
|
|
||||||
|
// minutes test
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - MINUTE * 2);
|
||||||
|
assert!(n.print_date().contains("minutes ago"));
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - MINUTE * 56);
|
||||||
|
assert!(n.print_date().contains("minutes ago"));
|
||||||
|
|
||||||
|
// hours test
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - HOUR);
|
||||||
|
assert!(n.print_date().contains("hours ago"));
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - HOUR * 23);
|
||||||
|
assert!(n.print_date().contains("hours ago"));
|
||||||
|
|
||||||
|
// days test
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - 2 * WEEK);
|
||||||
|
assert!(n.print_date().contains("days ago"));
|
||||||
|
|
||||||
|
// date test
|
||||||
|
n.received = OffsetDateTime::from_unix_timestamp(timestamp - 6 * WEEK);
|
||||||
|
let date = n.received.format("%d-%m-%y");
|
||||||
|
assert!(n.print_date().contains(&date))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -22,8 +22,8 @@ include!("../navbar/index.html"); .>
|
|||||||
<p class="notification__item-text"><.= notification.message .></p>
|
<p class="notification__item-text"><.= notification.message .></p>
|
||||||
<div class="notification-data__container">
|
<div class="notification-data__container">
|
||||||
<span class="notification__sender"><.= notification.name .></span>
|
<span class="notification__sender"><.= notification.name .></span>
|
||||||
<span>.</span>
|
<span>·</span>
|
||||||
<span class="notification__received"><.= notification.received .></span>
|
<span class="notification__received"><.= notification.print_date() .></span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</td>
|
</td>
|
||||||
|
|||||||
@@ -20,12 +20,12 @@
|
|||||||
.notification__table {
|
.notification__table {
|
||||||
background-color: $white;
|
background-color: $white;
|
||||||
width: 90%;
|
width: 90%;
|
||||||
|
padding: 0 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notification__title-text {
|
.notification__title-text {
|
||||||
padding-left: 10px;
|
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
padding: 0.75rem 1.25rem;
|
padding: 0.75rem 0.5rem;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|||||||
Reference in New Issue
Block a user