Add a page to list articles by tag
This commit is contained in:
		
							parent
							
								
									a54d2e9d71
								
							
						
					
					
						commit
						dd9c4a6a73
					
				| @ -59,6 +59,27 @@ impl Post { | ||||
|     find_by!(posts, find_by_slug, slug as String, blog_id as i32); | ||||
|     find_by!(posts, find_by_ap_url, ap_url as String); | ||||
| 
 | ||||
|     pub fn list_by_tag(conn: &PgConnection, tag: String, (min, max): (i32, i32)) -> Vec<Post> { | ||||
|         use schema::tags; | ||||
| 
 | ||||
|         let ids = tags::table.filter(tags::tag.eq(tag)).select(tags::post_id); | ||||
|         posts::table.filter(posts::id.eq(any(ids))) | ||||
|             .order(posts::creation_date.desc()) | ||||
|             .offset(min.into()) | ||||
|             .limit((max - min).into()) | ||||
|             .get_results::<Post>(conn) | ||||
|             .expect("Error loading posts by tag") | ||||
|     } | ||||
| 
 | ||||
|     pub fn count_for_tag(conn: &PgConnection, tag: String) -> i64 { | ||||
|         use schema::tags; | ||||
|         let ids = tags::table.filter(tags::tag.eq(tag)).select(tags::post_id); | ||||
|         posts::table.filter(posts::id.eq(any(ids))) | ||||
|             .count() | ||||
|             .get_result(conn) | ||||
|             .expect("Error counting posts by tag") | ||||
|     } | ||||
| 
 | ||||
|     pub fn count_local(conn: &PgConnection) -> usize { | ||||
|         use schema::post_authors; | ||||
|         use schema::users; | ||||
| @ -220,7 +241,7 @@ impl Post { | ||||
|         article.object_props.set_content_string(self.content.get().clone()).expect("Article::into_activity: content error"); | ||||
|         article.object_props.set_published_utctime(Utc.from_utc_datetime(&self.creation_date)).expect("Article::into_activity: published error"); | ||||
|         article.object_props.set_summary_string(self.subtitle.clone()).expect("Article::into_activity: summary error"); | ||||
|         article.object_props.tag = json!(mentions_json.append(&mut tags_json)); | ||||
|         article.object_props.tag = Some(json!(mentions_json.append(&mut tags_json))); | ||||
|         article.object_props.set_url_string(self.ap_url.clone()).expect("Article::into_activity: url error"); | ||||
|         article.object_props.set_to_link_vec::<Id>(to.into_iter().map(Id::new).collect()).expect("Article::into_activity: to error"); | ||||
|         article.object_props.set_cc_link_vec::<Id>(vec![]).expect("Article::into_activity: cc error"); | ||||
| @ -300,11 +321,11 @@ impl FromActivity<Article, PgConnection> for Post { | ||||
|             // save mentions and tags
 | ||||
|             if let Some(serde_json::Value::Array(tags)) = article.object_props.tag.clone() { | ||||
|                 for tag in tags.into_iter() { | ||||
|                     serde_json::from_value::<link::Mention>(tag) | ||||
|                     serde_json::from_value::<link::Mention>(tag.clone()) | ||||
|                         .map(|m| Mention::from_activity(conn, m, post.id, true)) | ||||
|                         .ok(); | ||||
| 
 | ||||
|                     serde_json::from_value::<Hashtag>(tag) | ||||
|                     serde_json::from_value::<Hashtag>(tag.clone()) | ||||
|                         .map(|t| Tag::from_activity(conn, t, post.id)) | ||||
|                         .ok(); | ||||
|                 } | ||||
|  | ||||
| @ -5,7 +5,7 @@ use ap_url; | ||||
| use instance::Instance; | ||||
| use schema::tags; | ||||
| 
 | ||||
| #[derive(Serialize, Queryable)] | ||||
| #[derive(Serialize, Queryable, Clone)] | ||||
| pub struct Tag { | ||||
|     pub id: i32, | ||||
|     pub tag: String, | ||||
| @ -24,12 +24,13 @@ pub struct NewTag { | ||||
| impl Tag { | ||||
|     insert!(tags, NewTag); | ||||
|     get!(tags); | ||||
|     find_by!(tags, find_by_name, tag as String); | ||||
|     list_by!(tags, for_post, post_id as i32); | ||||
| 
 | ||||
|     pub fn into_activity(&self, conn: &PgConnection) -> Hashtag { | ||||
|         let ht = Hashtag::default(); | ||||
|         let mut ht = Hashtag::default(); | ||||
|         ht.set_href_string(ap_url(format!("{}/tag/{}", Instance::get_local(conn).unwrap().public_domain, self.tag))).expect("Tag::into_activity: href error"); | ||||
|         ht.set_name_string(self.tag).expect("Tag::into_activity: name error"); | ||||
|         ht.set_name_string(self.tag.clone()).expect("Tag::into_activity: name error"); | ||||
|         ht | ||||
|     } | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/de.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/de.po
									
									
									
									
									
								
							| @ -545,5 +545,9 @@ msgstr "Über {{ instance_name }}" | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "Über {{ instance_name }}" | ||||
| 
 | ||||
| #~ msgid "Your password should be at least 8 characters long" | ||||
| #~ msgstr "Das Passwort sollte mindestens 8 Zeichen lang sein" | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/en.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/en.po
									
									
									
									
									
								
							| @ -531,3 +531,7 @@ msgstr "Welcome on {{ instance_name }}" | ||||
| 
 | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "Welcome on {{ instance_name }}" | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/fr.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/fr.po
									
									
									
									
									
								
							| @ -539,3 +539,7 @@ msgstr "Articles de {{ instance.name }}" | ||||
| 
 | ||||
| msgid "View all" | ||||
| msgstr "Tout afficher" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "Articles de {{ instance.name }}" | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/gl.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/gl.po
									
									
									
									
									
								
							| @ -535,3 +535,7 @@ msgstr "Acerca de {{ instance_name }}" | ||||
| 
 | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "Acerca de {{ instance_name }}" | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/nb.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/nb.po
									
									
									
									
									
								
							| @ -549,6 +549,10 @@ msgstr "Om {{ instance_name }}" | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "Om {{ instance_name }}" | ||||
| 
 | ||||
| #~ msgid "One reshare" | ||||
| #~ msgid_plural "{{ count }} reshares" | ||||
| #~ msgstr[0] "Én deling" | ||||
|  | ||||
							
								
								
									
										4
									
								
								po/pl.po
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								po/pl.po
									
									
									
									
									
								
							| @ -548,6 +548,10 @@ msgstr "O {{ instance_name }}" | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| #, fuzzy | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "O {{ instance_name }}" | ||||
| 
 | ||||
| #~ msgid "One reshare" | ||||
| #~ msgid_plural "{{ count }} reshares" | ||||
| #~ msgstr[0] "Jedno udostępnienie" | ||||
|  | ||||
| @ -519,3 +519,6 @@ msgstr "" | ||||
| 
 | ||||
| msgid "View all" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Articles tagged \"{{ tag }}\"" | ||||
| msgstr "" | ||||
|  | ||||
| @ -101,6 +101,9 @@ fn main() { | ||||
| 
 | ||||
|             routes::static_files, | ||||
| 
 | ||||
|             routes::tags::tag, | ||||
|             routes::tags::paginated_tag, | ||||
| 
 | ||||
|             routes::user::me, | ||||
|             routes::user::details, | ||||
|             routes::user::dashboard, | ||||
|  | ||||
| @ -108,6 +108,7 @@ pub mod notifications; | ||||
| pub mod posts; | ||||
| pub mod reshares; | ||||
| pub mod session; | ||||
| pub mod tags; | ||||
| pub mod user; | ||||
| pub mod well_known; | ||||
| 
 | ||||
|  | ||||
							
								
								
									
										28
									
								
								src/routes/tags.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/routes/tags.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | ||||
| use rocket_contrib::Template; | ||||
| use serde_json; | ||||
| 
 | ||||
| use plume_models::{ | ||||
|     db_conn::DbConn, | ||||
|     posts::Post, | ||||
|     tags::Tag, | ||||
|     users::User, | ||||
| }; | ||||
| use routes::Page; | ||||
| 
 | ||||
| #[get("/tag/<name>")] | ||||
| fn tag(user: Option<User>, conn: DbConn, name: String) -> Template { | ||||
|     paginated_tag(user, conn, name, Page::first()) | ||||
| } | ||||
| 
 | ||||
| #[get("/tag/<name>?<page>")] | ||||
| fn paginated_tag(user: Option<User>, conn: DbConn, name: String, page: Page) -> Template { | ||||
|     let tag = Tag::find_by_name(&*conn, name).expect("Rendering tags::tag: tag not found"); | ||||
|     let posts = Post::list_by_tag(&*conn, tag.tag.clone(), page.limits()); | ||||
|     Template::render("tags/index", json!({ | ||||
|         "tag": tag.clone(), | ||||
|         "account": user.map(|u| u.to_json(&*conn)), | ||||
|         "articles": posts.into_iter().map(|p| p.to_json(&*conn)).collect::<Vec<serde_json::Value>>(), | ||||
|         "page": page.page, | ||||
|         "n_pages": Page::total(Post::count_for_tag(&*conn, tag.tag) as i32) | ||||
|     })) | ||||
| } | ||||
| @ -36,7 +36,7 @@ | ||||
|         <p>{{ "This article is under the {{ license }} license." | _(license=article.post.license) }}</p> | ||||
|         <ul class="tags"> | ||||
|             {% for tag in article.tags %} | ||||
|                 <li>{{ tag.tag }}</li> | ||||
|                 <li><a href="/tag/{{ tag.tag }}">{{ tag.tag }}</a></li> | ||||
|             {% endfor %} | ||||
|         </ul> | ||||
|         <div class="flex"> | ||||
|  | ||||
							
								
								
									
										17
									
								
								templates/tags/index.html.tera
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								templates/tags/index.html.tera
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | ||||
| {% extends "base" %} | ||||
| {% import "macros" as macros %} | ||||
| 
 | ||||
| {% block title %} | ||||
| {{ 'Articles tagged "{{ tag }}"' | _(tag=tag.tag) }} | ||||
| {% endblock title %} | ||||
| 
 | ||||
| {% block content %} | ||||
|     <h1>{{ 'Articles tagged "{{ tag }}"' | _(tag=tag.tag) }}</h1> | ||||
| 
 | ||||
|     <div class="cards"> | ||||
|         {% for article in articles %} | ||||
|             {{ macros::post_card(article=article) }} | ||||
|         {% endfor %} | ||||
|     </div> | ||||
|     {{ macros::paginate(page=page, total=n_pages) }} | ||||
| {% endblock content %} | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user