2024-12-20 18:24:20 +03:00
|
|
|
use crate::components::editable_cell::EditableCell;
|
|
|
|
use crate::components::tag_editor::TagEditor;
|
2024-12-20 02:56:42 +03:00
|
|
|
use leptos::*;
|
|
|
|
use serde::Deserialize;
|
2024-12-20 18:24:20 +03:00
|
|
|
use uuid::Uuid;
|
2024-12-23 17:56:34 +03:00
|
|
|
use leptos::logging::log;
|
|
|
|
use crate::models::item::Item;
|
|
|
|
use std::sync::{Arc, Mutex};
|
2024-12-18 21:49:26 +03:00
|
|
|
|
|
|
|
#[derive(Deserialize, Clone, Debug)]
|
2024-12-20 18:24:20 +03:00
|
|
|
struct WikidataSuggestion {
|
|
|
|
id: String,
|
|
|
|
label: String,
|
|
|
|
description: Option<String>,
|
2024-12-18 21:49:26 +03:00
|
|
|
}
|
2024-12-20 02:56:42 +03:00
|
|
|
|
2024-12-06 14:45:14 +03:00
|
|
|
#[component]
|
2024-12-20 18:24:20 +03:00
|
|
|
pub fn ItemsList(
|
|
|
|
items: ReadSignal<Vec<Item>>,
|
|
|
|
set_items: WriteSignal<Vec<Item>>,
|
|
|
|
) -> impl IntoView {
|
2024-12-24 15:26:52 +03:00
|
|
|
|
|
|
|
// Ensure there's an initial empty row
|
|
|
|
set_items.set(vec![Item {
|
|
|
|
id: Uuid::new_v4().to_string(),
|
|
|
|
name: String::new(),
|
|
|
|
description: String::new(),
|
|
|
|
tags: vec![],
|
|
|
|
reviews: vec![],
|
|
|
|
wikidata_id: None,
|
|
|
|
}]);
|
|
|
|
|
2024-12-20 18:24:20 +03:00
|
|
|
let (wikidata_suggestions, set_wikidata_suggestions) =
|
|
|
|
create_signal(Vec::<WikidataSuggestion>::new());
|
2024-12-18 21:49:26 +03:00
|
|
|
|
2024-12-20 18:24:20 +03:00
|
|
|
// Fetch Wikidata suggestions
|
|
|
|
let fetch_wikidata_suggestions = move |query: String| {
|
2024-12-20 02:56:42 +03:00
|
|
|
spawn_local(async move {
|
2024-12-20 18:24:20 +03:00
|
|
|
if query.is_empty() {
|
2024-12-23 17:56:34 +03:00
|
|
|
set_wikidata_suggestions.set(Vec::new());
|
2024-12-20 18:24:20 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let url = format!(
|
|
|
|
"https://www.wikidata.org/w/api.php?action=wbsearchentities&search={}&language=en&limit=5&format=json&origin=*",
|
|
|
|
query
|
|
|
|
);
|
|
|
|
|
|
|
|
match gloo_net::http::Request::get(&url).send().await {
|
|
|
|
Ok(response) => {
|
|
|
|
if let Ok(data) = response.json::<WikidataResponse>().await {
|
2024-12-23 17:56:34 +03:00
|
|
|
set_wikidata_suggestions.set(data.search);
|
2024-12-18 21:49:26 +03:00
|
|
|
}
|
2024-12-20 18:24:20 +03:00
|
|
|
}
|
|
|
|
Err(_) => log!("Failed to fetch Wikidata suggestions"),
|
2024-12-18 21:49:26 +03:00
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
2024-12-20 02:56:42 +03:00
|
|
|
|
2024-12-20 18:24:20 +03:00
|
|
|
// Update item fields
|
|
|
|
let update_item = move |index: usize, field: &str, value: String| {
|
2024-12-20 02:56:42 +03:00
|
|
|
set_items.update(|items| {
|
|
|
|
if let Some(item) = items.get_mut(index) {
|
2024-12-20 18:24:20 +03:00
|
|
|
match field {
|
|
|
|
"name" => {
|
|
|
|
item.name = value.clone();
|
2024-12-24 15:26:52 +03:00
|
|
|
fetch_wikidata_suggestions(value.clone());
|
|
|
|
}
|
|
|
|
"description" => {
|
|
|
|
item.description = value.clone();
|
2024-12-20 02:56:42 +03:00
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
}
|
2024-12-18 16:00:40 +03:00
|
|
|
}
|
2024-12-24 15:26:52 +03:00
|
|
|
|
|
|
|
// Automatically add a new row when editing the last row
|
|
|
|
if index == items.len() - 1 && !value.is_empty() {
|
|
|
|
items.push(Item {
|
|
|
|
id: Uuid::new_v4().to_string(),
|
|
|
|
name: String::new(),
|
|
|
|
description: String::new(),
|
|
|
|
tags: vec![],
|
|
|
|
reviews: vec![],
|
|
|
|
wikidata_id: None,
|
|
|
|
});
|
|
|
|
}
|
2024-12-18 16:00:40 +03:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2024-12-20 18:24:20 +03:00
|
|
|
// Add a new tag to an item
|
|
|
|
let add_tag = move |index: usize, key: String, value: String| {
|
|
|
|
set_items.update(|items| {
|
|
|
|
if let Some(item) = items.get_mut(index) {
|
|
|
|
item.tags.push((key, value));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
// Remove a tag from an item
|
|
|
|
let remove_tag = move |item_index: usize, tag_index: usize| {
|
|
|
|
set_items.update(|items| {
|
|
|
|
if let Some(item) = items.get_mut(item_index) {
|
|
|
|
item.tags.remove(tag_index);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
// Remove an item
|
|
|
|
let remove_item = move |index: usize| {
|
|
|
|
set_items.update(|items| {
|
|
|
|
items.remove(index);
|
|
|
|
});
|
2024-12-20 02:56:42 +03:00
|
|
|
};
|
|
|
|
|
2024-12-06 14:45:14 +03:00
|
|
|
view! {
|
|
|
|
<div>
|
2024-12-20 18:24:20 +03:00
|
|
|
<h1>{ "Items List" }</h1>
|
2024-12-17 16:14:49 +03:00
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
2024-12-20 02:56:42 +03:00
|
|
|
<th>{ "Name" }</th>
|
2024-12-17 16:14:49 +03:00
|
|
|
<th>{ "Description" }</th>
|
|
|
|
<th>{ "Tags" }</th>
|
2024-12-20 18:24:20 +03:00
|
|
|
<th>{ "Actions" }</th>
|
2024-12-17 16:14:49 +03:00
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>
|
2024-12-20 18:24:20 +03:00
|
|
|
{move || items.get().iter().enumerate().map(|(index, item)| {
|
|
|
|
view! {
|
|
|
|
<tr>
|
|
|
|
// Editable Name Field with Wikidata Integration
|
|
|
|
<td>
|
2024-12-24 15:26:52 +03:00
|
|
|
<EditableCell
|
|
|
|
value=item.name.clone()
|
|
|
|
on_input=move |value| update_item(index, "name", value)
|
|
|
|
key=format!("name-{}", index) // Unique key per cell
|
|
|
|
/>
|
2024-12-20 18:24:20 +03:00
|
|
|
<ul>
|
|
|
|
{move || {
|
2024-12-23 17:56:34 +03:00
|
|
|
let suggestions = wikidata_suggestions.get().to_vec();
|
|
|
|
suggestions.into_iter().map(|suggestion| {
|
|
|
|
let label_for_click = suggestion.label.clone();
|
|
|
|
let label_for_display = suggestion.label.clone();
|
2024-12-23 22:54:51 +03:00
|
|
|
let description_for_click = suggestion.description.clone().unwrap_or_default();
|
|
|
|
let description_for_display = suggestion.description.clone().unwrap_or_default();
|
2024-12-23 17:56:34 +03:00
|
|
|
let id = suggestion.id.clone();
|
2024-12-23 22:54:51 +03:00
|
|
|
|
|
|
|
// Tags for the item
|
|
|
|
let tags = vec![
|
|
|
|
("source".to_string(), "wikidata".to_string()),
|
|
|
|
("wikidata_id".to_string(), id.clone()),
|
|
|
|
];
|
2024-12-24 15:26:52 +03:00
|
|
|
|
2024-12-20 18:24:20 +03:00
|
|
|
view! {
|
|
|
|
<li on:click=move |_| {
|
|
|
|
set_items.update(|items| {
|
|
|
|
if let Some(item) = items.get_mut(index) {
|
2024-12-24 14:27:32 +03:00
|
|
|
item.description = description_for_click.clone();
|
|
|
|
item.tags.extend(tags.clone());
|
2024-12-23 17:56:34 +03:00
|
|
|
item.wikidata_id = Some(id.clone());
|
2024-12-24 14:27:32 +03:00
|
|
|
item.name = label_for_click.clone();
|
2024-12-20 18:24:20 +03:00
|
|
|
}
|
|
|
|
});
|
2024-12-23 14:49:12 +03:00
|
|
|
}>
|
2024-12-24 14:27:32 +03:00
|
|
|
{ format!("{} - {}", label_for_display, description_for_display) }
|
2024-12-20 18:24:20 +03:00
|
|
|
</li>
|
2024-12-24 14:27:32 +03:00
|
|
|
}
|
|
|
|
}).collect::<Vec<_>>()
|
2024-12-20 18:24:20 +03:00
|
|
|
}}
|
|
|
|
</ul>
|
|
|
|
</td>
|
|
|
|
// Editable Description Field
|
|
|
|
<td>
|
2024-12-24 15:26:52 +03:00
|
|
|
<EditableCell
|
|
|
|
value=item.description.clone()
|
|
|
|
on_input=move |value| update_item(index, "description", value)
|
|
|
|
key=format!("description-{}", index)
|
|
|
|
/>
|
2024-12-20 18:24:20 +03:00
|
|
|
</td>
|
|
|
|
// Tag Editor
|
|
|
|
<td>
|
|
|
|
<TagEditor
|
|
|
|
tags=item.tags.clone()
|
|
|
|
on_add=move |key, value| add_tag(index, key, value)
|
2024-12-24 14:27:32 +03:00
|
|
|
on_remove=Arc::new(Mutex::new(move |tag_index: usize| remove_tag(index, tag_index)))
|
2024-12-20 18:24:20 +03:00
|
|
|
/>
|
|
|
|
</td>
|
|
|
|
// Actions
|
|
|
|
<td>
|
|
|
|
<button on:click=move |_| remove_item(index)>{ "Delete" }</button>
|
|
|
|
</td>
|
|
|
|
</tr>
|
|
|
|
}
|
|
|
|
}).collect::<Vec<_>>()}
|
2024-12-17 16:14:49 +03:00
|
|
|
</tbody>
|
|
|
|
</table>
|
2024-12-06 14:45:14 +03:00
|
|
|
</div>
|
|
|
|
}
|
2024-12-18 16:00:40 +03:00
|
|
|
}
|
2024-12-20 18:24:20 +03:00
|
|
|
|
|
|
|
#[derive(Deserialize, Clone, Debug)]
|
|
|
|
struct WikidataResponse {
|
|
|
|
search: Vec<WikidataSuggestion>,
|
2024-12-24 15:26:52 +03:00
|
|
|
}
|