add paginator

session-crate
Ondřej Hruška 4 years ago
parent 1f4af57b6e
commit f2a834c977
Signed by: MightyPork
GPG Key ID: 2C5FD5035250423D
  1. 67
      src/main.rs
  2. 11
      src/store/mod.rs
  3. 23
      templates/index.html.tera
  4. 54
      templates/static/style.css

@ -30,24 +30,41 @@ use indexmap::map::IndexMap;
pub struct ListContext<'a> {
pub fields: Vec<RenderedField<'a>>,
pub cards: Vec<RenderedCard<'a>>,
pub page: usize,
pub pages: usize,
}
#[get("/")]
fn route_index(store: State<RwLock<Store>>) -> Template {
#[get("/?<page>&<card>")]
fn route_index(store: State<RwLock<Store>>, page: Option<usize>, card: Option<usize>) -> Template {
let rg = store.read();
let per_page: usize = 20;
let mut page = page.unwrap_or_default();
let n_pages = (rg.data.cards.len() as f64 / per_page as f64).ceil() as usize;
if let Some(card_id) = card {
if let Some((n, _)) = rg.data.cards.iter().enumerate().find(|(_n, (id, _card))| **id == card_id) {
page = n / per_page;
}
}
let context = ListContext {
fields: render_empty_fields(&rg),
cards: rg.data.cards.iter().filter_map(|(id, card)| {
if let Value::Object(map) = card {
Some(RenderedCard {
fields: render_card_fields(&rg, map),
id: *id,
})
} else {
None
}
}).collect(),
pages: n_pages,
page,
cards: rg.data.cards.iter()
.skip(page * per_page)
.take(per_page)
.filter_map(|(id, card)| {
if let Value::Object(ref map) = card {
Some(RenderedCard {
fields: render_card_fields(&rg, map),
id: *id,
})
} else {
None
}
}).collect(),
};
Template::render("index", context)
@ -71,7 +88,7 @@ impl<'a> FromForm<'a> for MapFromForm {
}
}
fn collect_card_form(store : &Store, mut form : MapFromForm) -> IndexMap::<String, Value> {
fn collect_card_form(store: &Store, mut form: MapFromForm) -> IndexMap::<String, Value> {
let mut card = IndexMap::new();
for (k, field) in &store.model.fields {
@ -179,7 +196,7 @@ struct AddCardContext<'a> {
#[derive(Serialize)]
struct EditCardContext<'a> {
pub fields: Vec<RenderedField<'a>>,
pub id : usize,
pub id: usize,
}
#[get("/add")]
@ -198,16 +215,16 @@ fn route_add_save(form: Form<MapFromForm>, store: State<RwLock<Store>>) -> Redir
let mut rg = store.write();
let card = collect_card_form(&rg, form.into_inner());
rg.add_card(card);
let id = rg.add_card(card);
Redirect::found(uri!(route_index))
Redirect::found(uri!(route_index: page=_, card=id))
}
#[get("/edit/<id>")]
fn route_edit(id : usize, store: State<RwLock<Store>>) -> Option<Template> {
fn route_edit(id: usize, store: State<RwLock<Store>>) -> Option<Template> {
let rg = store.read();
if let Some(Value::Object(map)) = rg.data.cards.get(&id) {
if let Some(Value::Object(ref map)) = rg.data.cards.get(&id) {
let context = EditCardContext {
fields: render_card_fields(&rg, map),
id,
@ -220,13 +237,22 @@ fn route_edit(id : usize, store: State<RwLock<Store>>) -> Option<Template> {
}
#[post("/edit/<id>", data = "<form>")]
fn route_edit_save(id : usize, form: Form<MapFromForm>, store: State<RwLock<Store>>) -> Option<Redirect> {
fn route_edit_save(id: usize, form: Form<MapFromForm>, store: State<RwLock<Store>>) -> Option<Redirect> {
let mut rg = store.write();
let card = collect_card_form(&rg, form.into_inner());
rg.update_card(id, card);
Some(Redirect::found(uri!(route_index)))
Some(Redirect::found(uri!(route_index: page=_, card=id)))
}
#[get("/delete/<id>")]
fn route_delete(id: usize, store: State<RwLock<Store>>) -> Redirect {
let mut rg = store.write();
rg.delete_card(id);
Redirect::found(uri!(route_index: page=_, card=_))
}
fn main() {
@ -244,5 +270,6 @@ fn main() {
route_add_save,
route_edit,
route_edit_save,
route_delete,
]).launch();
}

@ -75,16 +75,17 @@ impl Store {
file.write(serialized.as_bytes()).expect("Error write data file");
}
pub fn add_card(&mut self, values : IndexMap::<String, Value>) {
pub fn add_card(&mut self, values : IndexMap::<String, Value>) -> usize {
let packed = serde_json::to_value(values).expect("Error serialize");
if let p @ Value::Object(_) = packed {
let id = self.data.counter;
self.data.counter += 1;
self.data.cards.insert(id, p);
self.persist();
id
} else {
panic!("Packing did not produce a map.");
}
self.persist()
}
pub fn update_card(&mut self, id : usize, values : IndexMap::<String, Value>) {
@ -102,6 +103,12 @@ impl Store {
self.persist()
}
pub fn delete_card(&mut self, id : usize) {
self.data.cards.remove(&id);
self.persist()
}
}
fn load_file(path: impl AsRef<Path>) -> String {

@ -22,7 +22,10 @@
<tbody>
{%- for card in cards %}
<tr>
<td><a href="/edit/{{card.id}}">Edit</a></td>
<td class="actions">
<a href="/edit/{{card.id}}">Edit</a>
<a href="/delete/{{card.id}}" onclick="return confirm('Confirm delete')">Delete</a>
</td>
{%- for field in card.fields %}
<td>
{%- if field.kind == "bool" -%}
@ -40,4 +43,22 @@
{% endfor %}
</tbody>
</table>
<nav class="paginate">
{% if pages > 1 %}
{% if page > 0 %}
<a href="/?page={{ page - 1 }}">«</a>
{% else %}
<span class="disabled">«</span>
{% endif %}
<span class="num" onclick="location.href='/?page='+(prompt('Page')-1);">{{ page + 1 }} of {{ pages }}</span>
{% if page < pages - 1 %}
<a href="/?page={{ page + 1 }}">»</a>
{% else %}
<span class="disabled">»</span>
{% endif %}
{% endif %}
</nav>
{%- endblock %}

@ -19,6 +19,16 @@ nav.top-nav {
border-bottom: 1px solid silver;
}
a {
color: gray;
text-decoration: none;
}
a:hover {
color: black;
text-decoration: underline;
}
nav.top-nav a {
display: inline-block;
padding: .75rem;
@ -112,6 +122,10 @@ textarea:focus,
margin-top: 1rem;
}
.cards-table .actions {
font-size: 90%;
}
.cards-table td,
.cards-table th {
padding: .5rem;
@ -128,3 +142,43 @@ textarea:focus,
.cards-table tbody tr:last-child td {
border-bottom: 2px solid silver;
}
.paginate {
margin: 1rem auto;
width: 300px;
text-align: center;
white-space: nowrap;
}
.paginate span,
.paginate a {
padding: .5rem 1rem;
border-radius: .5rem;
border: 1px solid silver;
text-decoration: none;
}
.paginate span.num {
border: 1px solid #ccc;
color: gray;
}
.paginate a {
cursor: pointer;
user-select: none;
}
.paginate a:hover {
background: #ccc;
text-decoration: none;
}
.paginate .disabled {
opacity: .5;
cursor: default;
}
.paginate .disabled:hover {
color: gray;
background: transparent;
}

Loading…
Cancel
Save