use crate::Route;
use gloo_net::http::Request;
use itertools::Itertools;
use lib::lib::*;
use stylist::style;
use wasm_bindgen::*;
use web_sys::{EventTarget, HtmlInputElement};
use yew::prelude::*;
use yew_router::scope_ext::RouterScopeExt;
#[derive(Properties, Clone, PartialEq, Eq)]
pub struct ResultComponentProps {
result: IndexedResource,
}
#[function_component(ResultComponent)]
fn result_component(props: &ResultComponentProps) -> Html {
let style = style!(
r#"
a {
text-decoration: none;
}
.underline-hover:hover {
text-decoration: underline;
}
"#
)
.unwrap();
let style = style.get_class_name().to_owned();
html! {
}
}
pub struct OSSE {
pub search_query: String,
pub results: Option, String>>, //TODO: some loading?
}
#[derive(Properties, PartialEq, Eq)]
pub struct OSSEProps {
pub api_endpoint: String,
pub initial_search_query: Option,
}
pub enum OSSEMessage {
SearchSubmitted,
SearchChanged(String),
SearchFinished(Result, String>),
}
impl Component for OSSE {
type Message = OSSEMessage;
type Properties = OSSEProps;
fn create(ctx: &Context) -> Self {
let mut search_query = String::from("");
//we push an SearchSubmitted message if inital_search_query is not none
if let Some(initial_search_query) = ctx.props().initial_search_query.clone() {
search_query = initial_search_query;
ctx.link().send_message(OSSEMessage::SearchSubmitted);
}
//WE may have data race between the future and the actual creation.
OSSE {
search_query: urlencoding::decode(search_query.as_str())
.to_owned()
.unwrap()
.to_string(),
results: None,
}
}
fn update(&mut self, ctx: &Context, msg: Self::Message) -> bool {
match msg {
OSSEMessage::SearchSubmitted => {
let api_endpoint = ctx.props().api_endpoint.clone();
let search_query = self.search_query.clone();
let navigator = ctx.link().navigator().unwrap();
navigator.push(&Route::OSSESearch {
query: urlencoding::encode(search_query.as_str()).to_string(),
});
ctx.link().send_future(async move {
let endpoint = format!("{}/search/{}", api_endpoint, search_query);
let fetched_response = match Request::get(endpoint.as_str()).send().await {
Ok(response) => response,
Err(_) => {
return OSSEMessage::SearchFinished(Err(
"Failed to connect to the API!".to_string(),
))
}
};
let fetched_results: Vec = match fetched_response.json().await
{
Err(_) => {
return OSSEMessage::SearchFinished(Err(
"Internal API Error!".to_string()
))
}
Ok(json) => json,
};
OSSEMessage::SearchFinished(Ok(fetched_results))
});
false
}
OSSEMessage::SearchChanged(search_query) => {
self.search_query = search_query;
true
}
OSSEMessage::SearchFinished(search_results) => {
match search_results {
Ok(results) => {
self.results = Some(Ok(results));
}
Err(error) => {
self.results = Some(Err(error));
}
};
true
}
}
}
fn view(&self, ctx: &Context) -> Html {
let onsubmit = ctx.link().callback(|event: SubmitEvent| {
event.prevent_default();
OSSEMessage::SearchSubmitted
});
let oninput = ctx.link().callback(|event: InputEvent| {
let target: EventTarget = event
.target()
.expect("Event should have a target when dispatched");
let input = target.unchecked_into::().value();
OSSEMessage::SearchChanged(input)
});
let display_results =
|maybe_results: &Option, String>>| -> Html {
if maybe_results.is_none() {
return html! {};
}
let results = maybe_results.as_ref().unwrap();
if results.is_err() {
return html! {
{format!("ERROR: {}", results.as_ref().err().unwrap())}
};
}
let results = results.as_ref().unwrap();
if !results.is_empty() {
results
.iter()
.sorted()
.map(|r| {
html! {
}
})
.collect::()
} else {
html! {
{"No results!"}
}
}
};
html! {
<>
//SET AT MIDDLE OF VIEWPORT IF NO SEARCHING AND TOP 25% IF SEARCHING
{"OSSE"}
{"Your favorite independent search engine."}
{display_results(&self.results)}
>
}
}
}
//Your favorite search engine in navbar
//Search in middle