2 Commits
v1.2.1 ... main

Author SHA1 Message Date
511e62b58c sticky pagination
All checks were successful
Build and Push Image / build-and-push (push) Successful in 4m6s
2025-12-28 15:20:21 -05:00
41aa78b672 added pagination controls
All checks were successful
Build and Push Image / build-and-push (push) Successful in 4m33s
2025-12-28 14:29:46 -05:00
13 changed files with 113 additions and 29 deletions

View File

@@ -21,6 +21,8 @@
top: 0;
position: sticky;
z-index: 100;
backdrop-filter: blur(20px);
box-shadow: 0 3px 10px $mainBGColor;
}
body {
@@ -195,3 +197,24 @@ form {
}
}
}
.pagination {
display: flex;
justify-content: center;
align-items: center;
gap: 5;
padding: 10px;
a {
transition: all 0.25s ease-out;
cursor: pointer;
user-select: none;
padding: 5px 10px;
background-color: $featureColor;
border-radius: 4px;
&:hover {
background-color: $focusColor;
}
}
}

View File

@@ -9,6 +9,8 @@ use crate::{
#[derive(PartialEq, Clone, Props)]
pub struct MediaGridProps {
pub query: Option<String>,
pub max_page: Signal<i32>,
pub total_items: Signal<i32>,
#[props(default = Some(1))]
pub page: Option<i32>,
#[props(default = Some(100))]
@@ -33,12 +35,13 @@ impl Into<PageFilter> for MediaGridProps {
}
#[component]
pub fn MediaGrid(props: MediaGridProps) -> Element {
pub fn MediaGrid(mut props: MediaGridProps) -> Element {
let media_result = use_resource(use_reactive!(|(props)| async move {
let mut client = get_rpc_client();
let result = client.list_media(props.into_request()).await;
if let Ok(items) = result {
return Ok(items.into_inner());
let res = items.into_inner();
return Ok(res);
} else {
let err = result.err().unwrap();
let message = err.message();
@@ -48,7 +51,13 @@ pub fn MediaGrid(props: MediaGridProps) -> Element {
match media_result.cloned() {
Some(value) => match value {
Ok(result) => rsx! {
Ok(result) => {
let pagination = result.pagination.unwrap();
let total_pages = pagination.total_pages;
let total_items = pagination.total_items;
props.max_page.set(total_pages.max(1));
props.total_items.set(total_items.max(1));
return rsx! {
div {
class: "mediaGrid",
// oncontextmenu: oncontext,
@@ -58,7 +67,8 @@ pub fn MediaGrid(props: MediaGridProps) -> Element {
}
})},
}
},
};
}
Err(msg) => rsx! {
div {
class: "mediaGrid",

View File

@@ -5,6 +5,7 @@ mod media_item;
mod metrics_token;
mod navbar;
mod notif;
mod pagination;
mod search;
pub use context_menu::*;
pub use media_grid::*;
@@ -12,5 +13,6 @@ pub use media_item::*;
pub use metrics_token::*;
pub use navbar::*;
pub use notif::*;
pub use pagination::*;
pub use search::*;
mod icons;

View File

@@ -0,0 +1,30 @@
use dioxus::prelude::*;
#[component]
pub fn Pagination(page: Signal<i32>, max_page: Signal<i32>, item_count: Signal<i32>) -> Element {
let cur_page_val = page.cloned();
let max_page_val = max_page.cloned();
let item_count_val = item_count.cloned();
rsx! {
div {
class: "pagination",
a {
onclick: move|_| page.set(1),
"First"
}
a {
onclick: move|_| page.set((cur_page_val - 1).max(1)),
"Prev"
}
div { "Page {cur_page_val} of {max_page_val} ({item_count_val} Media Items)" }
a {
onclick: move|_| page.set((cur_page_val + 1).min(max_page_val)),
"Next"
}
a {
onclick: move|_| page.set(max_page_val),
"Last"
}
}
}
}

View File

@@ -1,14 +1,14 @@
use dioxus::prelude::*;
#[component]
pub fn Search(query: Signal<String>) -> Element {
pub fn Search(query: Signal<String>, page: Signal<i32>) -> Element {
rsx! {
div { class: "searchBar stickyTop",
div { class: "searchBar",
input {
r#type: "search",
placeholder: "Search Files",
value: query,
oninput: move |event| query.set(event.value()),
oninput: move |event| {query.set(event.value()); page.set(1);},
}
}
}

View File

@@ -1,6 +1,6 @@
use crate::{
layouts::MainLayout,
views::{Home, Settings},
views::{Home, Media, Settings},
};
use dioxus::prelude::*;
@@ -8,8 +8,11 @@ use dioxus::prelude::*;
#[rustfmt::skip]
pub enum Route {
#[layout(MainLayout)]
#[route("/")]
Home { },
#[route("/media/:id")]
Media { id: String },
#[route("/settings")]
Settings {},
// #[end_layout]

View File

@@ -1,12 +1,18 @@
use crate::components::{MediaGrid, Search};
use crate::components::{MediaGrid, Pagination, Search};
use dioxus::prelude::*;
#[component]
pub fn Home() -> Element {
let query = use_signal(|| "".to_string());
let page = use_signal(|| 1 as i32);
let max_page = use_signal(|| 1 as i32);
let item_count = use_signal(|| 0 as i32);
rsx! {
Search { query }
MediaGrid { query: query.cloned() }
div {
class: "stickyTop",
Search { query, page },
Pagination { page, max_page, item_count },
}
MediaGrid { query: query.cloned(), page: page.cloned(), max_page, total_items: item_count }
}
}

View File

@@ -0,0 +1,8 @@
use dioxus::prelude::*;
#[component]
pub fn Media(id: String) -> Element {
rsx! {
{id}
}
}

View File

@@ -1,7 +1,9 @@
mod home;
mod login;
mod media;
pub use home::*;
pub use login::*;
pub use media::*;
mod settings;
pub use settings::Settings;

View File

@@ -5,13 +5,13 @@ using System.Text;
using System.Threading.Tasks;
namespace AobaCore.Models;
public class PagedResult<T>(List<T> items, int page, int pageSize, long totalItems)
public class PagedResult<T>(List<T> items, int page, int pageSize, int totalItems)
{
public List<T> Items { get; set; } = items;
public int Page { get; set; } = page;
public int PageSize { get; set; } = pageSize;
public long TotalItems { get; set; } = totalItems;
public long TotalPages { get; set; } = totalItems / pageSize;
public int TotalItems { get; set; } = totalItems;
public int TotalPages { get; set; } = (totalItems / pageSize) + 1;
public string? Query { get; set; }
}

View File

@@ -37,7 +37,7 @@ public class AobaService(IMongoDatabase db)
var total = await find.CountDocumentsAsync();
page -= 1;
var items = await find.Sort(sort).Skip(page * pageSize).Limit(pageSize).ToListAsync();
return new PagedResult<Media>(items, page, pageSize, total);
return new PagedResult<Media>(items, page, pageSize, (int)total);
}
public async Task<List<Media>> FindMediaWithExtAsync(string ext, CancellationToken cancellationToken = default)

View File

@@ -54,8 +54,8 @@ message ListResponse {
message Pagination {
int32 page = 1;
int32 pageSize = 2;
int64 totalPages = 3;
int64 totalItems = 4;
int32 totalPages = 3;
int32 totalItems = 4;
optional string query = 5;
}