All checks were successful
Build and Push Docker Image / docker (push) Successful in 1m19s
Events created with specific occurrence counts (like "repeat 5 times") were repeating forever instead of stopping after the specified number. Root cause: Frontend form collected recurrence_count and recurrence_until values correctly, but these weren't being passed through the event creation pipeline to the backend, which was hardcoding None values. Fix implemented across entire creation flow: 1. **Enhanced Parameter Conversion**: - Added recurrence_count and recurrence_until to to_create_event_params() tuple - Properly extracts values from form: recurrence_count, recurrence_until.map() 2. **Updated Backend Method Signature**: - Added recurrence_count: Option<u32> and recurrence_until: Option<String> - to create_event() method parameters 3. **Fixed Backend Implementation**: - Replace hardcoded None values with actual form parameters - "recurrence_end_date": recurrence_until, "recurrence_count": recurrence_count 4. **Updated Call Sites**: - Modified app.rs to pass params.18 (recurrence_count) and params.19 (recurrence_until) - Proper parameter indexing after tuple expansion Result: Complete recurrence control now works correctly: - ✅ Events with COUNT=5 stop after exactly 5 occurrences - ✅ Events with UNTIL date stop on specified date - ✅ Events with "repeat forever" continue indefinitely - ✅ Proper iCalendar RRULE generation with COUNT/UNTIL parameters 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
1194 lines
58 KiB
Rust
1194 lines
58 KiB
Rust
use crate::components::{
|
|
CalendarContextMenu, ContextMenu, CreateCalendarModal, CreateEventModal, DeleteAction,
|
|
EditAction, EventContextMenu, EventCreationData, RouteHandler, Sidebar, Theme, ViewMode,
|
|
};
|
|
use crate::components::sidebar::{Style};
|
|
use crate::models::ical::VEvent;
|
|
use crate::services::{calendar_service::UserInfo, CalendarService};
|
|
use chrono::NaiveDate;
|
|
use gloo_storage::{LocalStorage, Storage};
|
|
use wasm_bindgen::JsCast;
|
|
use web_sys::MouseEvent;
|
|
use yew::prelude::*;
|
|
use yew_router::prelude::*;
|
|
|
|
fn get_theme_event_colors() -> Vec<String> {
|
|
if let Some(window) = web_sys::window() {
|
|
if let Some(document) = window.document() {
|
|
if let Some(root) = document.document_element() {
|
|
if let Ok(Some(computed_style)) = window.get_computed_style(&root) {
|
|
if let Ok(colors_string) = computed_style.get_property_value("--event-colors") {
|
|
if !colors_string.is_empty() {
|
|
return colors_string
|
|
.split(',')
|
|
.map(|color| color.trim().to_string())
|
|
.filter(|color| !color.is_empty())
|
|
.collect();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vec![
|
|
"#3B82F6".to_string(),
|
|
"#10B981".to_string(),
|
|
"#F59E0B".to_string(),
|
|
"#EF4444".to_string(),
|
|
"#8B5CF6".to_string(),
|
|
"#06B6D4".to_string(),
|
|
"#84CC16".to_string(),
|
|
"#F97316".to_string(),
|
|
"#EC4899".to_string(),
|
|
"#6366F1".to_string(),
|
|
"#14B8A6".to_string(),
|
|
"#F3B806".to_string(),
|
|
"#8B5A2B".to_string(),
|
|
"#6B7280".to_string(),
|
|
"#DC2626".to_string(),
|
|
"#7C3AED".to_string(),
|
|
]
|
|
}
|
|
|
|
#[function_component]
|
|
pub fn App() -> Html {
|
|
let auth_token = use_state(|| -> Option<String> { LocalStorage::get("auth_token").ok() });
|
|
|
|
let user_info = use_state(|| -> Option<UserInfo> { None });
|
|
let color_picker_open = use_state(|| -> Option<String> { None });
|
|
let create_modal_open = use_state(|| false);
|
|
let context_menu_open = use_state(|| false);
|
|
let context_menu_pos = use_state(|| (0i32, 0i32));
|
|
let context_menu_calendar_path = use_state(|| -> Option<String> { None });
|
|
let event_context_menu_open = use_state(|| false);
|
|
let event_context_menu_pos = use_state(|| (0i32, 0i32));
|
|
let event_context_menu_event = use_state(|| -> Option<VEvent> { None });
|
|
let calendar_context_menu_open = use_state(|| false);
|
|
let calendar_context_menu_pos = use_state(|| (0i32, 0i32));
|
|
let calendar_context_menu_date = use_state(|| -> Option<NaiveDate> { None });
|
|
let create_event_modal_open = use_state(|| false);
|
|
let selected_date_for_event = use_state(|| -> Option<NaiveDate> { None });
|
|
let event_edit_scope = use_state(|| -> Option<EditAction> { None });
|
|
let _recurring_edit_modal_open = use_state(|| false);
|
|
let _recurring_edit_event = use_state(|| -> Option<VEvent> { None });
|
|
let _recurring_edit_data = use_state(|| -> Option<EventCreationData> { None });
|
|
|
|
// Calendar view state - load from localStorage if available
|
|
let current_view = use_state(|| {
|
|
// Try to load saved view mode from localStorage
|
|
if let Ok(saved_view) = LocalStorage::get::<String>("calendar_view_mode") {
|
|
match saved_view.as_str() {
|
|
"week" => ViewMode::Week,
|
|
_ => ViewMode::Month,
|
|
}
|
|
} else {
|
|
ViewMode::Month // Default to month view
|
|
}
|
|
});
|
|
|
|
// Theme state - load from localStorage if available
|
|
let current_theme = use_state(|| {
|
|
// Try to load saved theme from localStorage
|
|
if let Ok(saved_theme) = LocalStorage::get::<String>("calendar_theme") {
|
|
Theme::from_value(&saved_theme)
|
|
} else {
|
|
Theme::Default // Default theme
|
|
}
|
|
});
|
|
|
|
// Style state - load from localStorage if available
|
|
let current_style = use_state(|| {
|
|
// Try to load saved style from localStorage
|
|
if let Ok(saved_style) = LocalStorage::get::<String>("calendar_style") {
|
|
Style::from_value(&saved_style)
|
|
} else {
|
|
Style::Default // Default style
|
|
}
|
|
});
|
|
|
|
let available_colors = use_state(|| get_theme_event_colors());
|
|
|
|
let on_login = {
|
|
let auth_token = auth_token.clone();
|
|
Callback::from(move |token: String| {
|
|
auth_token.set(Some(token));
|
|
})
|
|
};
|
|
|
|
let on_logout = {
|
|
let auth_token = auth_token.clone();
|
|
let user_info = user_info.clone();
|
|
Callback::from(move |_| {
|
|
let _ = LocalStorage::delete("auth_token");
|
|
auth_token.set(None);
|
|
user_info.set(None);
|
|
})
|
|
};
|
|
|
|
let on_view_change = {
|
|
let current_view = current_view.clone();
|
|
Callback::from(move |new_view: ViewMode| {
|
|
// Save view mode to localStorage
|
|
let view_string = match new_view {
|
|
ViewMode::Month => "month",
|
|
ViewMode::Week => "week",
|
|
};
|
|
let _ = LocalStorage::set("calendar_view_mode", view_string);
|
|
|
|
// Update state
|
|
current_view.set(new_view);
|
|
})
|
|
};
|
|
|
|
let on_theme_change = {
|
|
let current_theme = current_theme.clone();
|
|
let available_colors = available_colors.clone();
|
|
Callback::from(move |new_theme: Theme| {
|
|
// Save theme to localStorage
|
|
let _ = LocalStorage::set("calendar_theme", new_theme.value());
|
|
|
|
// Apply theme to document root
|
|
if let Some(document) = web_sys::window().and_then(|w| w.document()) {
|
|
if let Some(root) = document.document_element() {
|
|
let _ = root.set_attribute("data-theme", new_theme.value());
|
|
}
|
|
}
|
|
|
|
// Update state
|
|
current_theme.set(new_theme);
|
|
|
|
// Update available colors after theme change
|
|
available_colors.set(get_theme_event_colors());
|
|
})
|
|
};
|
|
|
|
let on_style_change = {
|
|
let current_style = current_style.clone();
|
|
Callback::from(move |new_style: Style| {
|
|
// Save style to localStorage
|
|
let _ = LocalStorage::set("calendar_style", new_style.value());
|
|
|
|
// Hot-swap stylesheet
|
|
if let Some(window) = web_sys::window() {
|
|
if let Some(document) = window.document() {
|
|
// Remove existing style link if it exists
|
|
if let Some(existing_link) = document.get_element_by_id("dynamic-style") {
|
|
existing_link.remove();
|
|
}
|
|
|
|
// Create and append new stylesheet link only if style has a path
|
|
if let Some(stylesheet_path) = new_style.stylesheet_path() {
|
|
if let Ok(link) = document.create_element("link") {
|
|
let link = link.dyn_into::<web_sys::HtmlLinkElement>().unwrap();
|
|
link.set_id("dynamic-style");
|
|
link.set_rel("stylesheet");
|
|
link.set_href(stylesheet_path);
|
|
|
|
if let Some(head) = document.head() {
|
|
let _ = head.append_child(&link);
|
|
}
|
|
}
|
|
}
|
|
// If stylesheet_path is None (Default style), just removing the dynamic link is enough
|
|
}
|
|
}
|
|
|
|
// Update state
|
|
current_style.set(new_style);
|
|
})
|
|
};
|
|
|
|
// Apply initial theme on mount
|
|
{
|
|
let current_theme = current_theme.clone();
|
|
use_effect_with((), move |_| {
|
|
let theme = (*current_theme).clone();
|
|
if let Some(document) = web_sys::window().and_then(|w| w.document()) {
|
|
if let Some(root) = document.document_element() {
|
|
let _ = root.set_attribute("data-theme", theme.value());
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// Apply initial style on mount
|
|
{
|
|
let current_style = current_style.clone();
|
|
use_effect_with((), move |_| {
|
|
let style = (*current_style).clone();
|
|
if let Some(window) = web_sys::window() {
|
|
if let Some(document) = window.document() {
|
|
// Create and append stylesheet link for initial style only if it has a path
|
|
if let Some(stylesheet_path) = style.stylesheet_path() {
|
|
if let Ok(link) = document.create_element("link") {
|
|
let link = link.dyn_into::<web_sys::HtmlLinkElement>().unwrap();
|
|
link.set_id("dynamic-style");
|
|
link.set_rel("stylesheet");
|
|
link.set_href(stylesheet_path);
|
|
|
|
if let Some(head) = document.head() {
|
|
let _ = head.append_child(&link);
|
|
}
|
|
}
|
|
}
|
|
// If initial style is Default (None), no additional stylesheet needed
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
// Fetch user info when token is available
|
|
{
|
|
let user_info = user_info.clone();
|
|
let auth_token = auth_token.clone();
|
|
|
|
use_effect_with((*auth_token).clone(), move |token| {
|
|
if let Some(token) = token {
|
|
let user_info = user_info.clone();
|
|
let token = token.clone();
|
|
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
let password = if let Ok(credentials_str) =
|
|
LocalStorage::get::<String>("caldav_credentials")
|
|
{
|
|
if let Ok(credentials) =
|
|
serde_json::from_str::<serde_json::Value>(&credentials_str)
|
|
{
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
if !password.is_empty() {
|
|
match calendar_service.fetch_user_info(&token, &password).await {
|
|
Ok(mut info) => {
|
|
if let Ok(saved_colors_json) =
|
|
LocalStorage::get::<String>("calendar_colors")
|
|
{
|
|
if let Ok(saved_info) =
|
|
serde_json::from_str::<UserInfo>(&saved_colors_json)
|
|
{
|
|
for saved_cal in &saved_info.calendars {
|
|
for cal in &mut info.calendars {
|
|
if cal.path == saved_cal.path {
|
|
cal.color = saved_cal.color.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
user_info.set(Some(info));
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::log_1(
|
|
&format!("Failed to fetch user info: {}", err).into(),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
} else {
|
|
user_info.set(None);
|
|
}
|
|
|
|
|| ()
|
|
});
|
|
}
|
|
|
|
let on_outside_click = {
|
|
let color_picker_open = color_picker_open.clone();
|
|
let context_menu_open = context_menu_open.clone();
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
let calendar_context_menu_open = calendar_context_menu_open.clone();
|
|
Callback::from(move |e: MouseEvent| {
|
|
// Check if any context menu or color picker is open
|
|
let any_menu_open = color_picker_open.is_some()
|
|
|| *context_menu_open
|
|
|| *event_context_menu_open
|
|
|| *calendar_context_menu_open;
|
|
|
|
if any_menu_open {
|
|
// Prevent the default action and stop event propagation
|
|
e.prevent_default();
|
|
e.stop_propagation();
|
|
}
|
|
|
|
// Close all open menus/pickers
|
|
color_picker_open.set(None);
|
|
context_menu_open.set(false);
|
|
event_context_menu_open.set(false);
|
|
calendar_context_menu_open.set(false);
|
|
})
|
|
};
|
|
|
|
// Compute if any context menu is open
|
|
let any_context_menu_open = color_picker_open.is_some()
|
|
|| *context_menu_open
|
|
|| *event_context_menu_open
|
|
|| *calendar_context_menu_open;
|
|
|
|
let on_color_change = {
|
|
let user_info = user_info.clone();
|
|
let color_picker_open = color_picker_open.clone();
|
|
Callback::from(move |(calendar_path, color): (String, String)| {
|
|
if let Some(mut info) = (*user_info).clone() {
|
|
for calendar in &mut info.calendars {
|
|
if calendar.path == calendar_path {
|
|
calendar.color = color.clone();
|
|
break;
|
|
}
|
|
}
|
|
user_info.set(Some(info.clone()));
|
|
|
|
if let Ok(json) = serde_json::to_string(&info) {
|
|
let _ = LocalStorage::set("calendar_colors", json);
|
|
}
|
|
}
|
|
color_picker_open.set(None);
|
|
})
|
|
};
|
|
|
|
let on_color_picker_toggle = {
|
|
let color_picker_open = color_picker_open.clone();
|
|
Callback::from(move |calendar_path: String| {
|
|
if color_picker_open.as_ref() == Some(&calendar_path) {
|
|
color_picker_open.set(None);
|
|
} else {
|
|
color_picker_open.set(Some(calendar_path));
|
|
}
|
|
})
|
|
};
|
|
|
|
let on_calendar_context_menu = {
|
|
let context_menu_open = context_menu_open.clone();
|
|
let context_menu_pos = context_menu_pos.clone();
|
|
let context_menu_calendar_path = context_menu_calendar_path.clone();
|
|
Callback::from(move |(event, calendar_path): (MouseEvent, String)| {
|
|
context_menu_open.set(true);
|
|
context_menu_pos.set((event.client_x(), event.client_y()));
|
|
context_menu_calendar_path.set(Some(calendar_path));
|
|
})
|
|
};
|
|
|
|
let on_event_context_menu = {
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
let event_context_menu_pos = event_context_menu_pos.clone();
|
|
let event_context_menu_event = event_context_menu_event.clone();
|
|
Callback::from(move |(event, calendar_event): (MouseEvent, VEvent)| {
|
|
event_context_menu_open.set(true);
|
|
event_context_menu_pos.set((event.client_x(), event.client_y()));
|
|
event_context_menu_event.set(Some(calendar_event));
|
|
})
|
|
};
|
|
|
|
let on_calendar_date_context_menu = {
|
|
let calendar_context_menu_open = calendar_context_menu_open.clone();
|
|
let calendar_context_menu_pos = calendar_context_menu_pos.clone();
|
|
let calendar_context_menu_date = calendar_context_menu_date.clone();
|
|
Callback::from(move |(event, date): (MouseEvent, NaiveDate)| {
|
|
calendar_context_menu_open.set(true);
|
|
calendar_context_menu_pos.set((event.client_x(), event.client_y()));
|
|
calendar_context_menu_date.set(Some(date));
|
|
})
|
|
};
|
|
|
|
let on_create_event_click = {
|
|
let create_event_modal_open = create_event_modal_open.clone();
|
|
let selected_date_for_event = selected_date_for_event.clone();
|
|
let calendar_context_menu_date = calendar_context_menu_date.clone();
|
|
Callback::from(move |_: MouseEvent| {
|
|
create_event_modal_open.set(true);
|
|
selected_date_for_event.set((*calendar_context_menu_date).clone());
|
|
})
|
|
};
|
|
|
|
let on_event_create = {
|
|
let create_event_modal_open = create_event_modal_open.clone();
|
|
let auth_token = auth_token.clone();
|
|
Callback::from(move |event_data: EventCreationData| {
|
|
// Check if this is an update operation (has original_uid) or a create operation
|
|
if let Some(original_uid) = event_data.original_uid.clone() {
|
|
web_sys::console::log_1(&format!("Updating event via modal: {:?}", event_data).into());
|
|
|
|
create_event_modal_open.set(false);
|
|
|
|
// Handle the update operation using the existing backend update logic
|
|
if let Some(token) = (*auth_token).clone() {
|
|
let event_data_for_update = event_data.clone();
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
// Get CalDAV password from storage
|
|
let password = if let Ok(credentials_str) =
|
|
LocalStorage::get::<String>("caldav_credentials")
|
|
{
|
|
if let Ok(credentials) =
|
|
serde_json::from_str::<serde_json::Value>(&credentials_str)
|
|
{
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
// Convert EventCreationData to update parameters
|
|
let params = event_data_for_update.to_create_event_params();
|
|
|
|
// Determine if this is a recurring event update
|
|
let is_recurring = matches!(event_data_for_update.recurrence, crate::components::event_form::RecurrenceType::Daily |
|
|
crate::components::event_form::RecurrenceType::Weekly |
|
|
crate::components::event_form::RecurrenceType::Monthly |
|
|
crate::components::event_form::RecurrenceType::Yearly);
|
|
|
|
let update_result = if is_recurring && event_data_for_update.edit_scope.is_some() {
|
|
// Use series update endpoint for recurring events
|
|
let edit_action = event_data_for_update.edit_scope.unwrap();
|
|
let scope = match edit_action {
|
|
crate::components::EditAction::EditAll => "all_in_series".to_string(),
|
|
crate::components::EditAction::EditFuture => "this_and_future".to_string(),
|
|
crate::components::EditAction::EditThis => "this_only".to_string(),
|
|
};
|
|
|
|
calendar_service
|
|
.update_series(
|
|
&token,
|
|
&password,
|
|
original_uid.clone(),
|
|
params.0, // title
|
|
params.1, // description
|
|
params.2, // start_date
|
|
params.3, // start_time
|
|
params.4, // end_date
|
|
params.5, // end_time
|
|
params.6, // location
|
|
params.7, // all_day
|
|
params.8, // status
|
|
params.9, // class
|
|
params.10, // priority
|
|
params.11, // organizer
|
|
params.12, // attendees
|
|
params.13, // categories
|
|
params.14, // reminder
|
|
params.15, // recurrence
|
|
params.17, // calendar_path (skipping recurrence_days)
|
|
scope,
|
|
event_data_for_update.occurrence_date.map(|d| d.format("%Y-%m-%d").to_string()), // occurrence_date
|
|
)
|
|
.await
|
|
} else {
|
|
// Use regular update endpoint for single events
|
|
calendar_service
|
|
.update_event(
|
|
&token,
|
|
&password,
|
|
original_uid.clone(),
|
|
params.0, // title
|
|
params.1, // description
|
|
params.2, // start_date
|
|
params.3, // start_time
|
|
params.4, // end_date
|
|
params.5, // end_time
|
|
params.6, // location
|
|
params.7, // all_day
|
|
params.8, // status
|
|
params.9, // class
|
|
params.10, // priority
|
|
params.11, // organizer
|
|
params.12, // attendees
|
|
params.13, // categories
|
|
params.14, // reminder
|
|
params.15, // recurrence
|
|
params.16, // recurrence_days
|
|
params.17, // calendar_path
|
|
vec![], // exception_dates - empty for simple updates
|
|
None, // update_action - None for regular updates
|
|
None, // until_date - None for regular updates
|
|
)
|
|
.await
|
|
};
|
|
|
|
match update_result {
|
|
Ok(_) => {
|
|
web_sys::console::log_1(&"Event updated successfully via modal".into());
|
|
// Trigger a page reload to refresh events from all calendars
|
|
if let Some(window) = web_sys::window() {
|
|
let _ = window.location().reload();
|
|
}
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::error_1(
|
|
&format!("Failed to update event: {}", err).into(),
|
|
);
|
|
web_sys::window()
|
|
.unwrap()
|
|
.alert_with_message(&format!("Failed to update event: {}", err))
|
|
.unwrap();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
|
|
web_sys::console::log_1(&format!("Creating event: {:?}", event_data).into());
|
|
|
|
// Save the selected calendar as the last used calendar
|
|
if let Some(ref calendar_path) = event_data.selected_calendar {
|
|
let _ = LocalStorage::set("last_used_calendar", calendar_path);
|
|
|
|
// Also sync to backend asynchronously
|
|
let calendar_path_for_sync = calendar_path.clone();
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let preferences_service = crate::services::preferences::PreferencesService::new();
|
|
if let Err(e) = preferences_service.update_last_used_calendar(&calendar_path_for_sync).await {
|
|
web_sys::console::warn_1(&format!("Failed to sync last used calendar to backend: {}", e).into());
|
|
}
|
|
});
|
|
}
|
|
|
|
create_event_modal_open.set(false);
|
|
|
|
if let Some(_token) = (*auth_token).clone() {
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let _calendar_service = CalendarService::new();
|
|
|
|
// Get CalDAV password from storage
|
|
let _password = if let Ok(credentials_str) =
|
|
LocalStorage::get::<String>("caldav_credentials")
|
|
{
|
|
if let Ok(credentials) =
|
|
serde_json::from_str::<serde_json::Value>(&credentials_str)
|
|
{
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
let params = event_data.to_create_event_params();
|
|
let create_result = _calendar_service
|
|
.create_event(
|
|
&_token, &_password, params.0, // title
|
|
params.1, // description
|
|
params.2, // start_date
|
|
params.3, // start_time
|
|
params.4, // end_date
|
|
params.5, // end_time
|
|
params.6, // location
|
|
params.7, // all_day
|
|
params.8, // status
|
|
params.9, // class
|
|
params.10, // priority
|
|
params.11, // organizer
|
|
params.12, // attendees
|
|
params.13, // categories
|
|
params.14, // reminder
|
|
params.15, // recurrence
|
|
params.16, // recurrence_days
|
|
params.18, // recurrence_count
|
|
params.19, // recurrence_until
|
|
params.17, // calendar_path
|
|
)
|
|
.await;
|
|
match create_result {
|
|
Ok(_) => {
|
|
web_sys::console::log_1(&"Event created successfully".into());
|
|
// Trigger a page reload to refresh events from all calendars
|
|
// TODO: This could be improved to do a more targeted refresh
|
|
web_sys::window().unwrap().location().reload().unwrap();
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::error_1(
|
|
&format!("Failed to create event: {}", err).into(),
|
|
);
|
|
web_sys::window()
|
|
.unwrap()
|
|
.alert_with_message(&format!("Failed to create event: {}", err))
|
|
.unwrap();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
})
|
|
};
|
|
|
|
let on_event_update = {
|
|
let auth_token = auth_token.clone();
|
|
Callback::from(
|
|
move |(
|
|
original_event,
|
|
new_start,
|
|
new_end,
|
|
preserve_rrule,
|
|
until_date,
|
|
update_scope,
|
|
occurrence_date,
|
|
): (
|
|
VEvent,
|
|
chrono::NaiveDateTime,
|
|
chrono::NaiveDateTime,
|
|
bool,
|
|
Option<chrono::DateTime<chrono::Utc>>,
|
|
Option<String>,
|
|
Option<String>,
|
|
)| {
|
|
web_sys::console::log_1(
|
|
&format!(
|
|
"Updating event: {} to new times: {} - {}",
|
|
original_event.uid,
|
|
new_start.format("%Y-%m-%d %H:%M"),
|
|
new_end.format("%Y-%m-%d %H:%M")
|
|
)
|
|
.into(),
|
|
);
|
|
|
|
// Use the original UID for all updates
|
|
let backend_uid = original_event.uid.clone();
|
|
|
|
if let Some(token) = (*auth_token).clone() {
|
|
let original_event = original_event.clone();
|
|
let backend_uid = backend_uid.clone();
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
// Get CalDAV password from storage
|
|
let password = if let Ok(credentials_str) =
|
|
LocalStorage::get::<String>("caldav_credentials")
|
|
{
|
|
if let Ok(credentials) =
|
|
serde_json::from_str::<serde_json::Value>(&credentials_str)
|
|
{
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
// Send local time directly to backend (backend will handle UTC conversion)
|
|
let start_date = new_start.format("%Y-%m-%d").to_string();
|
|
let start_time = new_start.format("%H:%M").to_string();
|
|
let end_date = new_end.format("%Y-%m-%d").to_string();
|
|
let end_time = new_end.format("%H:%M").to_string();
|
|
|
|
// Convert existing event data to string formats for the API
|
|
let status_str = match original_event.status {
|
|
Some(crate::models::ical::EventStatus::Tentative) => {
|
|
"TENTATIVE".to_string()
|
|
}
|
|
Some(crate::models::ical::EventStatus::Confirmed) => {
|
|
"CONFIRMED".to_string()
|
|
}
|
|
Some(crate::models::ical::EventStatus::Cancelled) => {
|
|
"CANCELLED".to_string()
|
|
}
|
|
None => "CONFIRMED".to_string(), // Default status
|
|
};
|
|
|
|
let class_str = match original_event.class {
|
|
Some(crate::models::ical::EventClass::Public) => "PUBLIC".to_string(),
|
|
Some(crate::models::ical::EventClass::Private) => "PRIVATE".to_string(),
|
|
Some(crate::models::ical::EventClass::Confidential) => {
|
|
"CONFIDENTIAL".to_string()
|
|
}
|
|
None => "PUBLIC".to_string(), // Default class
|
|
};
|
|
|
|
// Convert reminders to string format
|
|
let reminder_str = if !original_event.alarms.is_empty() {
|
|
// Convert from VAlarm to minutes before
|
|
"15".to_string() // TODO: Convert VAlarm trigger to minutes
|
|
} else {
|
|
"".to_string()
|
|
};
|
|
|
|
// Handle recurrence (keep existing)
|
|
let recurrence_str = original_event.rrule.unwrap_or_default();
|
|
let recurrence_days = vec![false; 7]; // Default - could be enhanced to parse existing recurrence
|
|
|
|
// Determine if this is a recurring event that needs series endpoint
|
|
let has_recurrence =
|
|
!recurrence_str.is_empty() && recurrence_str.to_uppercase() != "NONE";
|
|
|
|
let result = if let Some(scope) = update_scope.as_ref() {
|
|
// Use series endpoint for recurring event operations
|
|
if !has_recurrence {
|
|
web_sys::console::log_1(&"⚠️ Warning: update_scope provided for non-recurring event, using regular endpoint instead".into());
|
|
// Fall through to regular endpoint
|
|
None
|
|
} else {
|
|
Some(
|
|
calendar_service
|
|
.update_series(
|
|
&token,
|
|
&password,
|
|
backend_uid.clone(),
|
|
original_event.summary.clone().unwrap_or_default(),
|
|
original_event.description.clone().unwrap_or_default(),
|
|
start_date.clone(),
|
|
start_time.clone(),
|
|
end_date.clone(),
|
|
end_time.clone(),
|
|
original_event.location.clone().unwrap_or_default(),
|
|
original_event.all_day,
|
|
status_str.clone(),
|
|
class_str.clone(),
|
|
original_event.priority,
|
|
original_event
|
|
.organizer
|
|
.as_ref()
|
|
.map(|o| o.cal_address.clone())
|
|
.unwrap_or_default(),
|
|
original_event
|
|
.attendees
|
|
.iter()
|
|
.map(|a| a.cal_address.clone())
|
|
.collect::<Vec<_>>()
|
|
.join(","),
|
|
original_event.categories.join(","),
|
|
reminder_str.clone(),
|
|
recurrence_str.clone(),
|
|
original_event.calendar_path.clone(),
|
|
scope.clone(),
|
|
occurrence_date,
|
|
)
|
|
.await,
|
|
)
|
|
}
|
|
} else {
|
|
None
|
|
};
|
|
|
|
let result = if let Some(series_result) = result {
|
|
series_result
|
|
} else {
|
|
// Use regular endpoint
|
|
calendar_service
|
|
.update_event(
|
|
&token,
|
|
&password,
|
|
backend_uid,
|
|
original_event.summary.unwrap_or_default(),
|
|
original_event.description.unwrap_or_default(),
|
|
start_date,
|
|
start_time,
|
|
end_date,
|
|
end_time,
|
|
original_event.location.unwrap_or_default(),
|
|
original_event.all_day,
|
|
status_str,
|
|
class_str,
|
|
original_event.priority,
|
|
original_event
|
|
.organizer
|
|
.as_ref()
|
|
.map(|o| o.cal_address.clone())
|
|
.unwrap_or_default(),
|
|
original_event
|
|
.attendees
|
|
.iter()
|
|
.map(|a| a.cal_address.clone())
|
|
.collect::<Vec<_>>()
|
|
.join(","),
|
|
original_event.categories.join(","),
|
|
reminder_str,
|
|
recurrence_str,
|
|
recurrence_days,
|
|
original_event.calendar_path,
|
|
original_event.exdate.clone(),
|
|
if preserve_rrule {
|
|
Some("update_series".to_string())
|
|
} else {
|
|
Some("this_and_future".to_string())
|
|
},
|
|
until_date,
|
|
)
|
|
.await
|
|
};
|
|
|
|
match result {
|
|
Ok(_) => {
|
|
web_sys::console::log_1(&"Event updated successfully".into());
|
|
// Add small delay before reload to let any pending requests complete
|
|
wasm_bindgen_futures::spawn_local(async {
|
|
gloo_timers::future::sleep(std::time::Duration::from_millis(
|
|
100,
|
|
))
|
|
.await;
|
|
web_sys::window().unwrap().location().reload().unwrap();
|
|
});
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::error_1(
|
|
&format!("Failed to update event: {}", err).into(),
|
|
);
|
|
web_sys::window()
|
|
.unwrap()
|
|
.alert_with_message(&format!("Failed to update event: {}", err))
|
|
.unwrap();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},
|
|
)
|
|
};
|
|
|
|
let refresh_calendars = {
|
|
let auth_token = auth_token.clone();
|
|
let user_info = user_info.clone();
|
|
Callback::from(move |_| {
|
|
if let Some(token) = (*auth_token).clone() {
|
|
let user_info = user_info.clone();
|
|
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
let password = if let Ok(credentials_str) =
|
|
LocalStorage::get::<String>("caldav_credentials")
|
|
{
|
|
if let Ok(credentials) =
|
|
serde_json::from_str::<serde_json::Value>(&credentials_str)
|
|
{
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
match calendar_service.fetch_user_info(&token, &password).await {
|
|
Ok(mut info) => {
|
|
if let Ok(saved_colors_json) =
|
|
LocalStorage::get::<String>("calendar_colors")
|
|
{
|
|
if let Ok(saved_info) =
|
|
serde_json::from_str::<UserInfo>(&saved_colors_json)
|
|
{
|
|
for saved_cal in &saved_info.calendars {
|
|
for cal in &mut info.calendars {
|
|
if cal.path == saved_cal.path {
|
|
cal.color = saved_cal.color.clone();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
user_info.set(Some(info));
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::log_1(
|
|
&format!("Failed to refresh calendars: {}", err).into(),
|
|
);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
})
|
|
};
|
|
|
|
// Debug logging
|
|
web_sys::console::log_1(
|
|
&format!("App rendering: auth_token = {:?}", auth_token.is_some()).into(),
|
|
);
|
|
|
|
html! {
|
|
<BrowserRouter>
|
|
<div class="app" onclick={on_outside_click}>
|
|
{
|
|
if auth_token.is_some() {
|
|
html! {
|
|
<>
|
|
<Sidebar
|
|
user_info={(*user_info).clone()}
|
|
on_logout={on_logout}
|
|
on_create_calendar={Callback::from({
|
|
let create_modal_open = create_modal_open.clone();
|
|
move |_| create_modal_open.set(true)
|
|
})}
|
|
color_picker_open={(*color_picker_open).clone()}
|
|
on_color_change={on_color_change}
|
|
on_color_picker_toggle={on_color_picker_toggle}
|
|
available_colors={(*available_colors).clone()}
|
|
on_calendar_context_menu={on_calendar_context_menu}
|
|
current_view={(*current_view).clone()}
|
|
on_view_change={on_view_change}
|
|
current_theme={(*current_theme).clone()}
|
|
on_theme_change={on_theme_change}
|
|
current_style={(*current_style).clone()}
|
|
on_style_change={on_style_change}
|
|
/>
|
|
<main class="app-main">
|
|
<RouteHandler
|
|
auth_token={(*auth_token).clone()}
|
|
user_info={(*user_info).clone()}
|
|
on_login={on_login.clone()}
|
|
on_event_context_menu={Some(on_event_context_menu.clone())}
|
|
on_calendar_context_menu={Some(on_calendar_date_context_menu.clone())}
|
|
view={(*current_view).clone()}
|
|
on_create_event_request={Some(on_event_create.clone())}
|
|
on_event_update_request={Some(on_event_update.clone())}
|
|
context_menus_open={any_context_menu_open}
|
|
/>
|
|
</main>
|
|
</>
|
|
}
|
|
} else {
|
|
html! {
|
|
<div class="login-layout">
|
|
<RouteHandler
|
|
auth_token={(*auth_token).clone()}
|
|
user_info={(*user_info).clone()}
|
|
on_login={on_login.clone()}
|
|
on_event_context_menu={Some(on_event_context_menu.clone())}
|
|
on_calendar_context_menu={Some(on_calendar_date_context_menu.clone())}
|
|
on_event_update_request={Some(on_event_update.clone())}
|
|
on_create_event_request={Some(on_event_create.clone())}
|
|
context_menus_open={any_context_menu_open}
|
|
/>
|
|
</div>
|
|
}
|
|
}
|
|
}
|
|
|
|
<CreateCalendarModal
|
|
is_open={*create_modal_open}
|
|
on_close={Callback::from({
|
|
let create_modal_open = create_modal_open.clone();
|
|
move |_| create_modal_open.set(false)
|
|
})}
|
|
on_create={Callback::from({
|
|
let auth_token = auth_token.clone();
|
|
let refresh_calendars = refresh_calendars.clone();
|
|
let create_modal_open = create_modal_open.clone();
|
|
move |(name, description, color): (String, Option<String>, Option<String>)| {
|
|
if let Some(token) = (*auth_token).clone() {
|
|
let refresh_calendars = refresh_calendars.clone();
|
|
let create_modal_open = create_modal_open.clone();
|
|
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
let password = if let Ok(credentials_str) = LocalStorage::get::<String>("caldav_credentials") {
|
|
if let Ok(credentials) = serde_json::from_str::<serde_json::Value>(&credentials_str) {
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
match calendar_service.create_calendar(&token, &password, name, description, color).await {
|
|
Ok(_) => {
|
|
web_sys::console::log_1(&"Calendar created successfully!".into());
|
|
refresh_calendars.emit(());
|
|
create_modal_open.set(false);
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::log_1(&format!("Failed to create calendar: {}", err).into());
|
|
create_modal_open.set(false);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
})}
|
|
available_colors={available_colors.iter().map(|c| c.to_string()).collect::<Vec<_>>()}
|
|
/>
|
|
|
|
<ContextMenu
|
|
is_open={*context_menu_open}
|
|
x={context_menu_pos.0}
|
|
y={context_menu_pos.1}
|
|
on_close={Callback::from({
|
|
let context_menu_open = context_menu_open.clone();
|
|
move |_| context_menu_open.set(false)
|
|
})}
|
|
on_delete={Callback::from({
|
|
let auth_token = auth_token.clone();
|
|
let context_menu_calendar_path = context_menu_calendar_path.clone();
|
|
let refresh_calendars = refresh_calendars.clone();
|
|
move |_: MouseEvent| {
|
|
if let (Some(token), Some(calendar_path)) = ((*auth_token).clone(), (*context_menu_calendar_path).clone()) {
|
|
let refresh_calendars = refresh_calendars.clone();
|
|
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
let password = if let Ok(credentials_str) = LocalStorage::get::<String>("caldav_credentials") {
|
|
if let Ok(credentials) = serde_json::from_str::<serde_json::Value>(&credentials_str) {
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
match calendar_service.delete_calendar(&token, &password, calendar_path).await {
|
|
Ok(_) => {
|
|
web_sys::console::log_1(&"Calendar deleted successfully!".into());
|
|
refresh_calendars.emit(());
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::log_1(&format!("Failed to delete calendar: {}", err).into());
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
})}
|
|
/>
|
|
|
|
<EventContextMenu
|
|
is_open={*event_context_menu_open}
|
|
x={event_context_menu_pos.0}
|
|
y={event_context_menu_pos.1}
|
|
event={(*event_context_menu_event).clone()}
|
|
on_close={Callback::from({
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
move |_| event_context_menu_open.set(false)
|
|
})}
|
|
on_edit={Callback::from({
|
|
let _event_context_menu_event = event_context_menu_event.clone();
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
let create_event_modal_open = create_event_modal_open.clone();
|
|
let event_edit_scope = event_edit_scope.clone();
|
|
move |edit_action: EditAction| {
|
|
// Set the edit scope and close the context menu
|
|
event_edit_scope.set(Some(edit_action));
|
|
event_context_menu_open.set(false);
|
|
create_event_modal_open.set(true);
|
|
}
|
|
})}
|
|
on_delete={Callback::from({
|
|
let auth_token = auth_token.clone();
|
|
let event_context_menu_event = event_context_menu_event.clone();
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
let refresh_calendars = refresh_calendars.clone();
|
|
move |delete_action: DeleteAction| {
|
|
if let (Some(token), Some(event)) = ((*auth_token).clone(), (*event_context_menu_event).clone()) {
|
|
let _refresh_calendars = refresh_calendars.clone();
|
|
let event_context_menu_open = event_context_menu_open.clone();
|
|
|
|
// Log the delete action for now - we'll implement different behaviors later
|
|
match delete_action {
|
|
DeleteAction::DeleteThis => web_sys::console::log_1(&"Delete this event".into()),
|
|
DeleteAction::DeleteFollowing => web_sys::console::log_1(&"Delete following events".into()),
|
|
DeleteAction::DeleteSeries => web_sys::console::log_1(&"Delete entire series".into()),
|
|
}
|
|
|
|
wasm_bindgen_futures::spawn_local(async move {
|
|
let calendar_service = CalendarService::new();
|
|
|
|
let password = if let Ok(credentials_str) = LocalStorage::get::<String>("caldav_credentials") {
|
|
if let Ok(credentials) = serde_json::from_str::<serde_json::Value>(&credentials_str) {
|
|
credentials["password"].as_str().unwrap_or("").to_string()
|
|
} else {
|
|
String::new()
|
|
}
|
|
} else {
|
|
String::new()
|
|
};
|
|
|
|
if let (Some(calendar_path), Some(event_href)) = (&event.calendar_path, &event.href) {
|
|
// Convert delete action to string and get occurrence date
|
|
let action_str = match delete_action {
|
|
DeleteAction::DeleteThis => "delete_this".to_string(),
|
|
DeleteAction::DeleteFollowing => "delete_following".to_string(),
|
|
DeleteAction::DeleteSeries => "delete_series".to_string(),
|
|
};
|
|
|
|
// Get the occurrence date from the clicked event
|
|
let occurrence_date = Some(event.dtstart.date_naive().format("%Y-%m-%d").to_string());
|
|
|
|
web_sys::console::log_1(&format!("🔄 Delete action: {}", action_str).into());
|
|
web_sys::console::log_1(&format!("🔄 Event UID: {}", event.uid).into());
|
|
web_sys::console::log_1(&format!("🔄 Event start: {}", event.dtstart).into());
|
|
web_sys::console::log_1(&format!("🔄 Occurrence date: {:?}", occurrence_date).into());
|
|
|
|
match calendar_service.delete_event(
|
|
&token,
|
|
&password,
|
|
calendar_path.clone(),
|
|
event_href.clone(),
|
|
action_str,
|
|
occurrence_date
|
|
).await {
|
|
Ok(message) => {
|
|
web_sys::console::log_1(&format!("Delete response: {}", message).into());
|
|
|
|
// Show the message to the user to explain what actually happened
|
|
if message.contains("Warning") {
|
|
web_sys::window().unwrap().alert_with_message(&message).unwrap();
|
|
}
|
|
|
|
// Close the context menu
|
|
event_context_menu_open.set(false);
|
|
// Force a page reload to refresh the calendar events
|
|
web_sys::window().unwrap().location().reload().unwrap();
|
|
}
|
|
Err(err) => {
|
|
web_sys::console::log_1(&format!("Failed to delete event: {}", err).into());
|
|
web_sys::window().unwrap().alert_with_message(&format!("Failed to delete event: {}", err)).unwrap();
|
|
}
|
|
}
|
|
} else {
|
|
web_sys::console::log_1(&"Missing calendar_path or href - cannot delete event".into());
|
|
}
|
|
});
|
|
}
|
|
}
|
|
})}
|
|
/>
|
|
|
|
<CalendarContextMenu
|
|
is_open={*calendar_context_menu_open}
|
|
x={calendar_context_menu_pos.0}
|
|
y={calendar_context_menu_pos.1}
|
|
on_close={Callback::from({
|
|
let calendar_context_menu_open = calendar_context_menu_open.clone();
|
|
move |_| calendar_context_menu_open.set(false)
|
|
})}
|
|
on_create_event={on_create_event_click}
|
|
/>
|
|
|
|
<CreateEventModal
|
|
is_open={*create_event_modal_open}
|
|
selected_date={(*selected_date_for_event).clone()}
|
|
initial_start_time={None}
|
|
initial_end_time={None}
|
|
event_to_edit={(*event_context_menu_event).clone()}
|
|
edit_scope={(*event_edit_scope).clone()}
|
|
on_close={Callback::from({
|
|
let create_event_modal_open = create_event_modal_open.clone();
|
|
let event_context_menu_event = event_context_menu_event.clone();
|
|
let event_edit_scope = event_edit_scope.clone();
|
|
move |_| {
|
|
create_event_modal_open.set(false);
|
|
// Clear the event being edited and edit scope
|
|
event_context_menu_event.set(None);
|
|
event_edit_scope.set(None);
|
|
}
|
|
})}
|
|
on_create={on_event_create}
|
|
available_calendars={user_info.as_ref().map(|ui| ui.calendars.clone()).unwrap_or_default()}
|
|
/>
|
|
</div>
|
|
</BrowserRouter>
|
|
}
|
|
}
|