consolidate viewdata & split apart templates
This commit is contained in:
parent
d199f1c0b7
commit
88d14fe501
3 changed files with 93 additions and 79 deletions
|
@ -167,6 +167,7 @@ pub struct FishMeta {
|
||||||
pub name_en: String,
|
pub name_en: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone)]
|
||||||
pub struct CombinedFish<'a> {
|
pub struct CombinedFish<'a> {
|
||||||
pub entry: &'a FishEntry,
|
pub entry: &'a FishEntry,
|
||||||
pub meta: &'a FishMeta,
|
pub meta: &'a FishMeta,
|
||||||
|
@ -176,7 +177,7 @@ pub struct CombinedFish<'a> {
|
||||||
pub rarity: f32,
|
pub rarity: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Clone, Copy)]
|
||||||
pub struct Window {
|
pub struct Window {
|
||||||
pub start_time: DateTime<Utc>,
|
pub start_time: DateTime<Utc>,
|
||||||
pub duration: Duration,
|
pub duration: Duration,
|
||||||
|
|
27
src/main.rs
27
src/main.rs
|
@ -7,7 +7,7 @@ use axum::{
|
||||||
routing::{get, post},
|
routing::{get, post},
|
||||||
Router,
|
Router,
|
||||||
};
|
};
|
||||||
use data::{Data, Filters};
|
use data::{CombinedFish, Data, Filters};
|
||||||
use maud::{html, Markup};
|
use maud::{html, Markup};
|
||||||
use nanoid::nanoid;
|
use nanoid::nanoid;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
|
@ -81,7 +81,30 @@ async fn main_handler(
|
||||||
|
|
||||||
// If this is a HTMX-sent request, don't resend the entire page, just the list.
|
// If this is a HTMX-sent request, don't resend the entire page, just the list.
|
||||||
let is_htmx = headers.contains_key("HX-Request");
|
let is_htmx = headers.contains_key("HX-Request");
|
||||||
Ok(templates::main_page(state, caught_fish, acc_id, &filters, is_htmx).into_response())
|
|
||||||
|
// Extract the list of fish we want to render.
|
||||||
|
let meta = state.data.fish_with_meta();
|
||||||
|
let mut values: Vec<&CombinedFish> = filters.filter(meta.values().collect(), &caught_fish);
|
||||||
|
values.sort_by(|afish, bfish| {
|
||||||
|
bfish
|
||||||
|
.is_up
|
||||||
|
.cmp(&afish.is_up)
|
||||||
|
.then(bfish.is_always_up.cmp(&afish.is_always_up))
|
||||||
|
.then(bfish.rarity.total_cmp(&afish.rarity).reverse())
|
||||||
|
.then(bfish.meta.name_en.cmp(&afish.meta.name_en))
|
||||||
|
});
|
||||||
|
|
||||||
|
let state = state.clone();
|
||||||
|
let filters = filters.clone();
|
||||||
|
Ok(templates::main_page(templates::ViewData {
|
||||||
|
state,
|
||||||
|
fish: values,
|
||||||
|
caught_fish,
|
||||||
|
acc_id,
|
||||||
|
filters,
|
||||||
|
only_list: is_htmx,
|
||||||
|
})
|
||||||
|
.into_response())
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn insert_cf_handler(
|
async fn insert_cf_handler(
|
||||||
|
|
110
src/templates.rs
110
src/templates.rs
|
@ -9,6 +9,15 @@ use crate::{
|
||||||
AppState,
|
AppState,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
pub struct ViewData<'a> {
|
||||||
|
pub state: State<Arc<AppState>>,
|
||||||
|
pub fish: Vec<&'a CombinedFish<'a>>,
|
||||||
|
pub caught_fish: Vec<i32>,
|
||||||
|
pub acc_id: String,
|
||||||
|
pub filters: Filters,
|
||||||
|
pub only_list: bool,
|
||||||
|
}
|
||||||
|
|
||||||
pub fn layout(content: Markup) -> Markup {
|
pub fn layout(content: Markup) -> Markup {
|
||||||
html! {
|
html! {
|
||||||
(DOCTYPE)
|
(DOCTYPE)
|
||||||
|
@ -31,45 +40,42 @@ pub fn layout(content: Markup) -> Markup {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn main_page(
|
pub fn main_page(data: ViewData) -> Markup {
|
||||||
state: State<Arc<AppState>>,
|
|
||||||
caught_fish: Vec<i32>,
|
|
||||||
acc_id: String,
|
|
||||||
filters: &Filters,
|
|
||||||
only_list: bool,
|
|
||||||
) -> Markup {
|
|
||||||
let meta = state.data.fish_with_meta();
|
|
||||||
let mut values: Vec<&CombinedFish> = filters.filter(meta.values().collect(), &caught_fish);
|
|
||||||
|
|
||||||
values.sort_by(|afish, bfish| {
|
|
||||||
bfish
|
|
||||||
.is_up
|
|
||||||
.cmp(&afish.is_up)
|
|
||||||
.then(bfish.is_always_up.cmp(&afish.is_always_up))
|
|
||||||
.then(bfish.rarity.total_cmp(&afish.rarity).reverse())
|
|
||||||
.then(bfish.meta.name_en.cmp(&afish.meta.name_en))
|
|
||||||
});
|
|
||||||
// values.sort_by(|afish, bfish| {
|
|
||||||
// if !afish.is_up && !bfish.is_up && !afish.windows.is_empty() && !bfish.windows.is_empty() {
|
|
||||||
// bfish
|
|
||||||
// .windows
|
|
||||||
// .first()
|
|
||||||
// .unwrap()
|
|
||||||
// .start_time
|
|
||||||
// .cmp(&afish.windows.first().unwrap().start_time)
|
|
||||||
// .reverse()
|
|
||||||
// } else {
|
|
||||||
// Ordering::Equal
|
|
||||||
// }
|
|
||||||
// });
|
|
||||||
let list = html! {
|
let list = html! {
|
||||||
h2.clock { "ET " (clock::get_current_eorzea_date().format("%H:%M")) }
|
h2.clock { "ET " (clock::get_current_eorzea_date().format("%H:%M")) }
|
||||||
@for fish in values {
|
(fish_list(&data))
|
||||||
|
script src="/static/scripts/dates.js" type="text/javascript" {}
|
||||||
|
};
|
||||||
|
|
||||||
|
let template = html! {
|
||||||
|
span style="display: none;" id="account-id" { (data.acc_id) }
|
||||||
|
(header(&data))
|
||||||
|
div id="list" hx-get="" hx-trigger="every 10s" hx-swap="innerHTML" hx-target="this" hx-on="changeDates" {
|
||||||
|
(list)
|
||||||
|
}
|
||||||
|
|
||||||
|
script src="/static/scripts/save.js" type="text/javascript" {}
|
||||||
|
};
|
||||||
|
|
||||||
|
if data.only_list {
|
||||||
|
list
|
||||||
|
} else {
|
||||||
|
layout(html! {
|
||||||
|
main {
|
||||||
|
(template)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn fish_list(data: &ViewData) -> Markup {
|
||||||
|
html! {
|
||||||
|
@for fish in data.fish.clone() {
|
||||||
section.fish.up[fish.is_up].alwaysup[fish.is_always_up] {
|
section.fish.up[fish.is_up].alwaysup[fish.is_always_up] {
|
||||||
.title {
|
.title {
|
||||||
div {
|
div {
|
||||||
@if !caught_fish.contains(&(fish.entry.id as i32)) {
|
@if !data.caught_fish.contains(&(fish.entry.id as i32)) {
|
||||||
form action=(format!("/{}/catch/{}", acc_id, fish.entry.id)) method="post" {
|
form action=(format!("/{}/catch/{}", data.acc_id, fish.entry.id)) method="post" {
|
||||||
button.catch-button type="submit" { (PreEscaped("✓")) }
|
button.catch-button type="submit" { (PreEscaped("✓")) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -116,15 +122,15 @@ pub fn main_page(
|
||||||
}
|
}
|
||||||
.how {
|
.how {
|
||||||
@for item_id in &fish.entry.best_catch_path {
|
@for item_id in &fish.entry.best_catch_path {
|
||||||
@if let Some(item) = state.data.db_data.items.get(item_id) {
|
@if let Some(item) = data.state.data.db_data.items.get(item_id) {
|
||||||
span.catchpath title=(item.name_en) {
|
span.catchpath title=(item.name_en) {
|
||||||
img src=(item.get_icon_url()) width="35";
|
img src=(item.get_icon_url()) width="35";
|
||||||
|
|
||||||
@if let Some(hookset) = item.get_hookset(&state.data) {
|
@if let Some(hookset) = item.get_hookset(&data.state.data) {
|
||||||
img.hookset src=(hookset) width="20";
|
img.hookset src=(hookset) width="20";
|
||||||
}
|
}
|
||||||
|
|
||||||
@if let Some(tug) = item.get_tug(&state.data) {
|
@if let Some(tug) = item.get_tug(&data.state.data) {
|
||||||
span.tug { (tug) }
|
span.tug { (tug) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -157,12 +163,11 @@ pub fn main_page(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
script src="/static/scripts/dates.js" type="text/javascript" {}
|
pub fn header(data: &ViewData) -> Markup {
|
||||||
};
|
html! {
|
||||||
|
|
||||||
let template = html! {
|
|
||||||
span style="display: none;" id="account-id" { (acc_id) }
|
|
||||||
.header {
|
.header {
|
||||||
div {}
|
div {}
|
||||||
.side {
|
.side {
|
||||||
|
@ -176,7 +181,7 @@ pub fn main_page(
|
||||||
summary { "Filters" }
|
summary { "Filters" }
|
||||||
form {
|
form {
|
||||||
fieldset {
|
fieldset {
|
||||||
@if filters.non_big_fish {
|
@if data.filters.non_big_fish {
|
||||||
input name="big" id="big" type="checkbox" checked;
|
input name="big" id="big" type="checkbox" checked;
|
||||||
} @else {
|
} @else {
|
||||||
input name="big" id="big" type="checkbox";
|
input name="big" id="big" type="checkbox";
|
||||||
|
@ -185,7 +190,7 @@ pub fn main_page(
|
||||||
}
|
}
|
||||||
|
|
||||||
fieldset {
|
fieldset {
|
||||||
@if filters.include_caught {
|
@if data.filters.include_caught {
|
||||||
input name="caught" id="caught" type="checkbox" checked;
|
input name="caught" id="caught" type="checkbox" checked;
|
||||||
} @else {
|
} @else {
|
||||||
input name="caught" id="caught" type="checkbox";
|
input name="caught" id="caught" type="checkbox";
|
||||||
|
@ -198,7 +203,7 @@ pub fn main_page(
|
||||||
br;
|
br;
|
||||||
select name="patches" id="patches" multiple {
|
select name="patches" id="patches" multiple {
|
||||||
@for patch in data::PATCHES {
|
@for patch in data::PATCHES {
|
||||||
@if filters.patches.contains(&patch) {
|
@if data.filters.patches.contains(&patch) {
|
||||||
option value=(patch) selected { (format!("{:.1}", patch)) }
|
option value=(patch) selected { (format!("{:.1}", patch)) }
|
||||||
} @else {
|
} @else {
|
||||||
option value=(patch) { (format!("{:.1}", patch)) }
|
option value=(patch) { (format!("{:.1}", patch)) }
|
||||||
|
@ -212,21 +217,6 @@ pub fn main_page(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
div id="list" hx-get="" hx-trigger="every 10s" hx-swap="innerHTML" hx-target="this" hx-on="changeDates" {
|
|
||||||
(list)
|
|
||||||
}
|
|
||||||
|
|
||||||
script src="/static/scripts/save.js" type="text/javascript" {}
|
|
||||||
};
|
|
||||||
|
|
||||||
if only_list {
|
|
||||||
list
|
|
||||||
} else {
|
|
||||||
layout(html! {
|
|
||||||
main {
|
|
||||||
(template)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue