feat(labels): add state to track when labels are being fetched(in progress)
This commit is contained in:
parent
ef7245b716
commit
40bb35d6a8
1 changed files with 230 additions and 178 deletions
|
@ -143,6 +143,10 @@ pub fn ItemsList(
|
||||||
|
|
||||||
// State to manage property cache
|
// State to manage property cache
|
||||||
let (property_cache, set_property_cache) = create_signal(HashMap::<String, HashMap<String, String>>::new());
|
let (property_cache, set_property_cache) = create_signal(HashMap::<String, HashMap<String, String>>::new());
|
||||||
|
|
||||||
|
// State to track when labels are being fetched
|
||||||
|
let (is_fetching_labels, set_is_fetching_labels) = create_signal(false);
|
||||||
|
|
||||||
#[cfg(feature = "ssr")]
|
#[cfg(feature = "ssr")]
|
||||||
fn get_current_url() -> String {
|
fn get_current_url() -> String {
|
||||||
use leptos::use_context;
|
use leptos::use_context;
|
||||||
|
@ -206,7 +210,11 @@ pub fn ItemsList(
|
||||||
|
|
||||||
// Fetch labels for the custom properties
|
// Fetch labels for the custom properties
|
||||||
let property_ids = custom_props_clone;
|
let property_ids = custom_props_clone;
|
||||||
let labels = fetch_property_labels(property_ids).await;
|
let labels = fetch_property_labels(
|
||||||
|
property_ids,
|
||||||
|
set_property_labels.clone(),
|
||||||
|
move |value| set_is_fetching_labels.set(value),
|
||||||
|
).await;
|
||||||
set_property_labels.update(|labels_map| {
|
set_property_labels.update(|labels_map| {
|
||||||
for (key, value) in labels {
|
for (key, value) in labels {
|
||||||
labels_map.insert(key, value);
|
labels_map.insert(key, value);
|
||||||
|
@ -445,7 +453,11 @@ pub fn ItemsList(
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
if !missing_ids.is_empty() {
|
if !missing_ids.is_empty() {
|
||||||
let new_labels = fetch_property_labels(missing_ids).await;
|
let new_labels = fetch_property_labels(
|
||||||
|
missing_ids,
|
||||||
|
set_property_labels.clone(),
|
||||||
|
move |value| { set_is_fetching_labels.set(value) },
|
||||||
|
).await;
|
||||||
set_property_labels.update(|labels| {
|
set_property_labels.update(|labels| {
|
||||||
labels.extend(new_labels.clone());
|
labels.extend(new_labels.clone());
|
||||||
});
|
});
|
||||||
|
@ -456,7 +468,7 @@ pub fn ItemsList(
|
||||||
for binding in bindings {
|
for binding in bindings {
|
||||||
if let (Some(prop), Some(value_label)) = (
|
if let (Some(prop), Some(value_label)) = (
|
||||||
binding["propLabel"]["value"].as_str(),
|
binding["propLabel"]["value"].as_str(),
|
||||||
binding["valueLabel"]["value"].as_str()
|
binding["valueLabel"]["value"].as_str(),
|
||||||
) {
|
) {
|
||||||
let prop_id = prop.replace("http://www.wikidata.org/prop/", "");
|
let prop_id = prop.replace("http://www.wikidata.org/prop/", "");
|
||||||
if let Some(label) = property_labels.get().get(&prop_id).cloned() {
|
if let Some(label) = property_labels.get().get(&prop_id).cloned() {
|
||||||
|
@ -480,7 +492,12 @@ pub fn ItemsList(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn fetch_property_labels(property_ids: Vec<String>) -> HashMap<String, String> {
|
async fn fetch_property_labels(
|
||||||
|
property_ids: Vec<String>,
|
||||||
|
set_property_labels: WriteSignal<HashMap<String, String>>,
|
||||||
|
set_is_fetching_labels: impl Fn(bool) + 'static,
|
||||||
|
) -> HashMap<String, String> {
|
||||||
|
set_is_fetching_labels(true);
|
||||||
log!("Fetching property labels for properties: {:?}", property_ids);
|
log!("Fetching property labels for properties: {:?}", property_ids);
|
||||||
|
|
||||||
// Remove the "http://www.wikidata.org/prop/" prefix from property IDs
|
// Remove the "http://www.wikidata.org/prop/" prefix from property IDs
|
||||||
|
@ -532,7 +549,8 @@ pub fn ItemsList(
|
||||||
binding["prop"]["value"].as_str(),
|
binding["prop"]["value"].as_str(),
|
||||||
binding["propLabel"]["value"].as_str()
|
binding["propLabel"]["value"].as_str()
|
||||||
) {
|
) {
|
||||||
let prop_id = prop.split('/').last().unwrap_or("").to_string();
|
let prop_id =
|
||||||
|
prop.split('/').last().unwrap_or("").to_string();
|
||||||
result.insert(prop_id.clone(), label.to_string());
|
result.insert(prop_id.clone(), label.to_string());
|
||||||
log!("Processed binding {}: prop_id = {}, label = {}", i, prop_id, label);
|
log!("Processed binding {}: prop_id = {}, label = {}", i, prop_id, label);
|
||||||
} else {
|
} else {
|
||||||
|
@ -543,6 +561,7 @@ pub fn ItemsList(
|
||||||
log!("Warning: No bindings found in the response");
|
log!("Warning: No bindings found in the response");
|
||||||
}
|
}
|
||||||
log!("Fetched {} property labels", result.len());
|
log!("Fetched {} property labels", result.len());
|
||||||
|
set_is_fetching_labels(false);
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
|
@ -572,189 +591,210 @@ pub fn ItemsList(
|
||||||
let property_cache = property_cache.clone();
|
let property_cache = property_cache.clone();
|
||||||
let set_property_cache = set_property_cache.clone();
|
let set_property_cache = set_property_cache.clone();
|
||||||
Arc::new(move |property: String| {
|
Arc::new(move |property: String| {
|
||||||
// Normalize the property ID
|
// Normalize the property ID
|
||||||
let normalized_property = property.replace("http://www.wikidata.org/prop/", "");
|
let normalized_property = property.replace("http://www.wikidata.org/prop/", "");
|
||||||
let normalized_property_clone = normalized_property.clone();
|
let normalized_property_clone = normalized_property.clone();
|
||||||
|
|
||||||
// Check if label already exists
|
// Check if label already exists
|
||||||
if !property_labels.get().contains_key(&normalized_property) {
|
if !property_labels.get().contains_key(&normalized_property) {
|
||||||
spawn_local({
|
spawn_local({
|
||||||
let normalized_property = normalized_property.clone();
|
let normalized_property = normalized_property.clone();
|
||||||
let set_property_labels = set_property_labels.clone();
|
let set_property_labels = set_property_labels.clone();
|
||||||
async move {
|
async move {
|
||||||
let labels = fetch_property_labels(vec![normalized_property.clone()]).await;
|
let labels = fetch_property_labels(
|
||||||
set_property_labels.update(|map| {
|
vec![normalized_property.clone()],
|
||||||
map.extend(labels);
|
set_property_labels.clone(),
|
||||||
});
|
move |value| set_is_fetching_labels.set(value),
|
||||||
}
|
)
|
||||||
});
|
.await;
|
||||||
}
|
set_property_labels.update(|map| {
|
||||||
|
map.extend(labels);
|
||||||
// Check if property is already selected
|
});
|
||||||
if !selected_properties.get().contains_key(&normalized_property) && !normalized_property.is_empty() {
|
|
||||||
// Add property to selected properties
|
|
||||||
set_selected_properties.update(|selected| {
|
|
||||||
selected.insert(normalized_property.clone(), true);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Save the selected property to the database
|
|
||||||
spawn_local({
|
|
||||||
let current_url = Rc::clone(¤t_url);
|
|
||||||
let normalized_property = normalized_property_clone.clone();
|
|
||||||
async move {
|
|
||||||
let response = gloo_net::http::Request::post(
|
|
||||||
&format!("/api/urls/{}/properties", encode(¤t_url))
|
|
||||||
)
|
|
||||||
.json(&normalized_property)
|
|
||||||
.unwrap()
|
|
||||||
.send()
|
|
||||||
.await;
|
|
||||||
|
|
||||||
match response {
|
|
||||||
Ok(resp) => {
|
|
||||||
if resp.status() == 200 {
|
|
||||||
log!("Property saved successfully");
|
|
||||||
} else {
|
|
||||||
log!("Error saving property: {}", resp.status_text());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(err) => {
|
|
||||||
log!("Error saving property: {:?}", err);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
});
|
}
|
||||||
}
|
|
||||||
|
|
||||||
set_custom_properties.update(|props| {
|
// Check if property is already selected
|
||||||
if !props.contains(&normalized_property) && !normalized_property.is_empty() {
|
if !selected_properties.get().contains_key(&normalized_property) && !normalized_property.is_empty() {
|
||||||
props.push(normalized_property.clone());
|
// Add property to selected properties
|
||||||
|
|
||||||
//update the selected_properties state when a new property is added
|
|
||||||
set_selected_properties.update(|selected| {
|
set_selected_properties.update(|selected| {
|
||||||
selected.insert(normalized_property.clone(), true);
|
selected.insert(normalized_property.clone(), true);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Ensure the grid updates reactively
|
// Save the selected property to the database
|
||||||
set_items.update(|items| {
|
spawn_local({
|
||||||
for item in items {
|
let current_url = Rc::clone(¤t_url);
|
||||||
item.custom_properties.entry(normalized_property.clone()).or_insert_with(|| "".to_string());
|
let normalized_property = normalized_property_clone.clone();
|
||||||
|
async move {
|
||||||
|
let response = gloo_net::http::Request::post(
|
||||||
|
&format!("/api/urls/{}/properties", encode(¤t_url))
|
||||||
|
)
|
||||||
|
.json(&normalized_property)
|
||||||
|
.unwrap()
|
||||||
|
.send()
|
||||||
|
.await;
|
||||||
|
|
||||||
// Save the updated item to the database
|
match response {
|
||||||
let item_clone = item.clone();
|
Ok(resp) => {
|
||||||
spawn_local({
|
if resp.status() == 200 {
|
||||||
let current_url = Rc::clone(¤t_url);
|
log!("Property saved successfully");
|
||||||
async move {
|
} else {
|
||||||
save_item_to_db(item_clone, selected_properties, current_url.to_string()).await;
|
log!("Error saving property: {}", resp.status_text());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
Err(err) => {
|
||||||
|
log!("Error saving property: {:?}", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Use the property label from the property_labels signal
|
|
||||||
let property_label = property_labels.get().get(&normalized_property).cloned().unwrap_or_else(|| normalized_property.clone());
|
|
||||||
log!("Added property with label: {}", property_label);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
// Fetch the relevant value for each item and populate the corresponding cells
|
set_custom_properties.update(|props| {
|
||||||
set_items.update(|items| {
|
if !props.contains(&normalized_property) && !normalized_property.is_empty() {
|
||||||
for item in items {
|
props.push(normalized_property.clone());
|
||||||
if let Some(wikidata_id) = &item.wikidata_id {
|
|
||||||
let wikidata_id = wikidata_id.clone();
|
//update the selected_properties state when a new property is added
|
||||||
let set_fetched_properties = set_fetched_properties.clone();
|
set_selected_properties.update(|selected| {
|
||||||
let set_property_labels = set_property_labels.clone();
|
selected.insert(normalized_property.clone(), true);
|
||||||
let property_clone = property.clone();
|
});
|
||||||
spawn_local(async move {
|
|
||||||
let properties = fetch_item_properties(&wikidata_id, set_property_labels.clone(), property_cache.clone(), set_property_cache.clone(), property_labels.clone()).await;
|
// Ensure the grid updates reactively
|
||||||
// Update fetched properties and property labels
|
set_items.update(|items| {
|
||||||
set_fetched_properties.update(|fp| {
|
for item in items {
|
||||||
fp.insert(wikidata_id.clone(), properties.clone());
|
item.custom_properties.entry(normalized_property.clone()).or_insert_with(|| "".to_string());
|
||||||
});
|
|
||||||
set_property_labels.update(|pl| {
|
// Save the updated item to the database
|
||||||
for (key, value) in properties.iter() {
|
let item_clone = item.clone();
|
||||||
pl.entry(key.clone()).or_insert_with(|| value.clone());
|
spawn_local({
|
||||||
}
|
let current_url = Rc::clone(¤t_url);
|
||||||
});
|
async move {
|
||||||
if let Some(value) = properties.get(&property_clone) {
|
save_item_to_db(item_clone, selected_properties, current_url.to_string()).await;
|
||||||
set_items.update(|items| {
|
|
||||||
if let Some(item) = items.iter_mut().find(|item| item.wikidata_id.as_ref().unwrap() == &wikidata_id) {
|
|
||||||
item.custom_properties.insert(property_clone.clone(), value.clone());
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Use the property label from the property_labels signal
|
||||||
|
let property_label = property_labels.get().get(&normalized_property).cloned().unwrap_or_else(|| normalized_property.clone());
|
||||||
|
log!("Added property with label: {}", property_label);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
});
|
// Fetch the relevant value for each item and populate the corresponding cells
|
||||||
})};
|
set_items.update(|items| {
|
||||||
|
for item in items {
|
||||||
|
if let Some(wikidata_id) = &item.wikidata_id {
|
||||||
|
let wikidata_id = wikidata_id.clone();
|
||||||
|
let set_fetched_properties = set_fetched_properties.clone();
|
||||||
|
let set_property_labels = set_property_labels.clone();
|
||||||
|
let property_clone = property.clone();
|
||||||
|
spawn_local(async move {
|
||||||
|
let properties = fetch_item_properties(
|
||||||
|
&wikidata_id,
|
||||||
|
set_property_labels.clone(),
|
||||||
|
property_cache.clone(),
|
||||||
|
set_property_cache.clone(),
|
||||||
|
property_labels.clone(),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
|
// Update fetched properties and property labels
|
||||||
|
set_fetched_properties.update(|fp| {
|
||||||
|
fp.insert(wikidata_id.clone(), properties.clone());
|
||||||
|
});
|
||||||
|
set_property_labels.update(|pl| {
|
||||||
|
for (key, value) in properties.iter() {
|
||||||
|
pl.entry(key.clone()).or_insert_with(|| value.clone());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if let Some(value) = properties.get(&property_clone) {
|
||||||
|
set_items.update(|items| {
|
||||||
|
if let Some(item) = items.iter_mut().find(|item| item.wikidata_id.as_ref().unwrap() == &wikidata_id) {
|
||||||
|
item.custom_properties.insert(property_clone.clone(), value.clone());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
// Update item fields
|
// Update item fields
|
||||||
let update_item = {
|
let update_item = {
|
||||||
let set_items = set_items.clone();
|
let set_items = set_items.clone();
|
||||||
let current_url = Rc::clone(¤t_url);
|
let current_url = Rc::clone(¤t_url);
|
||||||
Arc::new(move |index: usize, field: &str, value: String| {
|
Arc::new(move |index: usize, field: &str, value: String| {
|
||||||
let set_items = set_items.clone();
|
let set_items = set_items.clone();
|
||||||
let current_url = Rc::clone(¤t_url);
|
let current_url = Rc::clone(¤t_url);
|
||||||
set_items.update(move|items| {
|
set_items.update(move|items| {
|
||||||
if let Some(item) = items.get_mut(index) {
|
if let Some(item) = items.get_mut(index) {
|
||||||
match field {
|
match field {
|
||||||
"name" => {
|
"name" => {
|
||||||
item.name = value.clone();
|
item.name = value.clone();
|
||||||
fetch_wikidata_suggestions(format!("name-{}", index), value.clone());
|
fetch_wikidata_suggestions(format!("name-{}", index), value.clone());
|
||||||
|
|
||||||
// Fetch Wikidata properties if the field is "name" and the item has a valid Wikidata ID
|
// Fetch Wikidata properties if the field is "name" and the item has a valid Wikidata ID
|
||||||
if !value.is_empty() {
|
if !value.is_empty() {
|
||||||
if let Some(wikidata_id) = &item.wikidata_id {
|
if let Some(wikidata_id) = &item.wikidata_id {
|
||||||
let wikidata_id = wikidata_id.clone();
|
let wikidata_id = wikidata_id.clone();
|
||||||
spawn_local(async move {
|
spawn_local(async move {
|
||||||
let properties = fetch_item_properties(&wikidata_id, set_property_labels.clone(), property_cache.clone(), set_property_cache.clone(), property_labels.clone()).await;
|
let properties = fetch_item_properties(
|
||||||
|
&wikidata_id,
|
||||||
|
set_property_labels.clone(),
|
||||||
|
property_cache.clone(),
|
||||||
|
set_property_cache.clone(),
|
||||||
|
property_labels.clone(),
|
||||||
|
)
|
||||||
|
.await;
|
||||||
log!("Fetched properties for index {}: {:?}", index, properties);
|
log!("Fetched properties for index {}: {:?}", index, properties);
|
||||||
});
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
"description" => {
|
||||||
"description" => {
|
item.description = value.clone();
|
||||||
item.description = value.clone();
|
}
|
||||||
}
|
_ => {
|
||||||
_ => {
|
// Update custom property
|
||||||
// Update custom property
|
|
||||||
item.custom_properties.insert(field.to_string(), value.clone());
|
item.custom_properties.insert(field.to_string(), value.clone());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Save the updated item to the database
|
// Save the updated item to the database
|
||||||
let item_clone = item.clone();
|
let item_clone = item.clone();
|
||||||
spawn_local({
|
spawn_local({
|
||||||
let current_url = Rc::clone(¤t_url);
|
let current_url = Rc::clone(¤t_url);
|
||||||
async move {
|
async move {
|
||||||
save_item_to_db(item_clone, selected_properties, current_url.to_string()).await;
|
save_item_to_db(item_clone, selected_properties, current_url.to_string()).await;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
// Automatically add a new row when editing the last row
|
// Automatically add a new row when editing the last row
|
||||||
if index == items.len() - 1 && !value.is_empty() {
|
if index == items.len() - 1 && !value.is_empty() {
|
||||||
let new_item = Item {
|
let new_item = Item {
|
||||||
id: Uuid::new_v4().to_string(),
|
id: Uuid::new_v4().to_string(),
|
||||||
name: String::new(),
|
name: String::new(),
|
||||||
description: String::new(),
|
description: String::new(),
|
||||||
// reviews: vec![],
|
// reviews: vec![],
|
||||||
wikidata_id: None,
|
wikidata_id: None,
|
||||||
custom_properties: HashMap::new(),
|
custom_properties: HashMap::new(),
|
||||||
};
|
};
|
||||||
items.push(new_item.clone());
|
items.push(new_item.clone());
|
||||||
|
|
||||||
// Save the new item to the database
|
// Save the new item to the database
|
||||||
spawn_local({
|
spawn_local({
|
||||||
let current_url = Rc::clone(¤t_url);
|
let current_url = Rc::clone(¤t_url);
|
||||||
async move {
|
async move {
|
||||||
save_item_to_db(new_item, selected_properties, current_url.to_string()).await;
|
save_item_to_db(new_item, selected_properties, current_url.to_string()).await;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
log!("Items updated: {:?}", items);
|
log!("Items updated: {:?}", items);
|
||||||
});
|
});
|
||||||
})};
|
})
|
||||||
|
};
|
||||||
|
|
||||||
// List of properties to display as rows
|
// List of properties to display as rows
|
||||||
let properties = vec!["Name", "Description"];
|
let properties = vec!["Name", "Description"];
|
||||||
|
@ -933,10 +973,20 @@ pub fn ItemsList(
|
||||||
let property_label = property_labels.get().get(&normalized_property).cloned().unwrap_or_else(|| normalized_property.clone());
|
let property_label = property_labels.get().get(&normalized_property).cloned().unwrap_or_else(|| normalized_property.clone());
|
||||||
log!("Rendering property: {} -> {}", normalized_property, property_label);
|
log!("Rendering property: {} -> {}", normalized_property, property_label);
|
||||||
let property_clone_for_button = normalized_property.clone();
|
let property_clone_for_button = normalized_property.clone();
|
||||||
|
let normalized_property_clone = normalized_property.clone();
|
||||||
view! {
|
view! {
|
||||||
<tr>
|
<tr>
|
||||||
<td>
|
<td>
|
||||||
{ property_label }
|
{move || {
|
||||||
|
let property_label = property_labels.get().get(&normalized_property).cloned();
|
||||||
|
let is_fetching = is_fetching_labels.get();
|
||||||
|
|
||||||
|
if is_fetching {
|
||||||
|
normalized_property.clone()
|
||||||
|
} else {
|
||||||
|
property_label.unwrap_or_else(|| normalized_property.clone())
|
||||||
|
}
|
||||||
|
}}
|
||||||
<button class="delete-property" on:click=move |_| {
|
<button class="delete-property" on:click=move |_| {
|
||||||
log!("Deleting property: {}", property_clone_for_button);
|
log!("Deleting property: {}", property_clone_for_button);
|
||||||
remove_property_clone(property_clone_for_button.clone());
|
remove_property_clone(property_clone_for_button.clone());
|
||||||
|
@ -953,9 +1003,11 @@ pub fn ItemsList(
|
||||||
});
|
});
|
||||||
}>{ "Delete" }</button>
|
}>{ "Delete" }</button>
|
||||||
</td>
|
</td>
|
||||||
|
let normalized_property = property.replace("http://www.wikidata.org/prop/", "");
|
||||||
|
let normalized_property_clone = normalized_property.clone();
|
||||||
{move || {
|
{move || {
|
||||||
let update_item_cell = Arc::clone(&update_item_inner);
|
let update_item_cell = Arc::clone(&update_item_inner);
|
||||||
let property_clone_for_cells = normalized_property.clone();
|
let property_clone_for_cells = normalized_property_clone.clone();
|
||||||
items.get().iter().enumerate().map(move |(index, item)| {
|
items.get().iter().enumerate().map(move |(index, item)| {
|
||||||
let update_item_cell = Arc::clone(&update_item_cell);
|
let update_item_cell = Arc::clone(&update_item_cell);
|
||||||
let property_clone_for_closure = property_clone_for_cells.clone();
|
let property_clone_for_closure = property_clone_for_cells.clone();
|
||||||
|
|
Loading…
Add table
Reference in a new issue