|
#![allow(clippy::too_many_arguments)] |
|
|
|
|
|
|
|
|
|
|
|
use crate::helpers::translate_key; |
|
use crate::{EDITOR, EDITOR_HANDLE, EDITOR_HAS_CRASHED, Error}; |
|
use editor::application::Editor; |
|
use editor::consts::FILE_SAVE_SUFFIX; |
|
use editor::messages::input_mapper::utility_types::input_keyboard::ModifierKeys; |
|
use editor::messages::input_mapper::utility_types::input_mouse::{EditorMouseState, ScrollDelta, ViewportBounds}; |
|
use editor::messages::portfolio::document::utility_types::document_metadata::LayerNodeIdentifier; |
|
use editor::messages::portfolio::document::utility_types::network_interface::ImportOrExport; |
|
use editor::messages::portfolio::utility_types::Platform; |
|
use editor::messages::prelude::*; |
|
use editor::messages::tool::tool_messages::tool_prelude::WidgetId; |
|
use graph_craft::document::NodeId; |
|
use graphene_std::raster::color::Color; |
|
use serde::Serialize; |
|
use serde_wasm_bindgen::{self, from_value}; |
|
use std::cell::RefCell; |
|
use std::sync::atomic::Ordering; |
|
use std::time::Duration; |
|
use wasm_bindgen::prelude::*; |
|
|
|
|
|
|
|
#[wasm_bindgen(js_name = setRandomSeed)] |
|
pub fn set_random_seed(seed: u64) { |
|
editor::application::set_uuid_seed(seed); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = wasmMemory)] |
|
pub fn wasm_memory() -> JsValue { |
|
wasm_bindgen::memory() |
|
} |
|
|
|
|
|
|
|
|
|
#[wasm_bindgen] |
|
#[derive(Clone)] |
|
pub struct EditorHandle { |
|
|
|
frontend_message_handler_callback: js_sys::Function, |
|
} |
|
|
|
|
|
|
|
impl EditorHandle { |
|
pub fn send_frontend_message_to_js_rust_proxy(&self, message: FrontendMessage) { |
|
self.send_frontend_message_to_js(message); |
|
} |
|
} |
|
|
|
#[wasm_bindgen] |
|
impl EditorHandle { |
|
#[wasm_bindgen(constructor)] |
|
pub fn new(frontend_message_handler_callback: js_sys::Function) -> Self { |
|
let editor = Editor::new(); |
|
let editor_handle = EditorHandle { frontend_message_handler_callback }; |
|
if EDITOR.with(|handle| handle.lock().ok().map(|mut guard| *guard = Some(editor))).is_none() { |
|
log::error!("Attempted to initialize the editor more than once"); |
|
} |
|
if EDITOR_HANDLE.with(|handle| handle.lock().ok().map(|mut guard| *guard = Some(editor_handle.clone()))).is_none() { |
|
log::error!("Attempted to initialize the editor handle more than once"); |
|
} |
|
editor_handle |
|
} |
|
|
|
|
|
fn dispatch<T: Into<Message>>(&self, message: T) { |
|
|
|
if EDITOR_HAS_CRASHED.load(Ordering::SeqCst) { |
|
return; |
|
} |
|
|
|
|
|
let frontend_messages = editor(|editor| editor.handle_message(message.into())); |
|
|
|
|
|
for message in frontend_messages.into_iter() { |
|
self.send_frontend_message_to_js(message); |
|
} |
|
} |
|
|
|
|
|
fn send_frontend_message_to_js(&self, mut message: FrontendMessage) { |
|
if let FrontendMessage::UpdateDocumentLayerStructure { data_buffer } = message { |
|
message = FrontendMessage::UpdateDocumentLayerStructureJs { data_buffer: data_buffer.into() }; |
|
} |
|
|
|
let message_type = message.to_discriminant().local_name(); |
|
|
|
let serializer = serde_wasm_bindgen::Serializer::new().serialize_large_number_types_as_bigints(true); |
|
let message_data = message.serialize(&serializer).expect("Failed to serialize FrontendMessage"); |
|
|
|
let js_return_value = self.frontend_message_handler_callback.call2(&JsValue::null(), &JsValue::from(message_type), &message_data); |
|
|
|
if let Err(error) = js_return_value { |
|
error!( |
|
"While handling FrontendMessage \"{:?}\", JavaScript threw an error: {:?}", |
|
message.to_discriminant().local_name(), |
|
error, |
|
) |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
#[wasm_bindgen(js_name = initAfterFrontendReady)] |
|
pub fn init_after_frontend_ready(&self, platform: String) { |
|
|
|
let platform = match platform.as_str() { |
|
"Windows" => Platform::Windows, |
|
"Mac" => Platform::Mac, |
|
"Linux" => Platform::Linux, |
|
_ => Platform::Unknown, |
|
}; |
|
self.dispatch(GlobalsMessage::SetPlatform { platform }); |
|
self.dispatch(Message::Init); |
|
|
|
|
|
{ |
|
let f = std::rc::Rc::new(RefCell::new(None)); |
|
let g = f.clone(); |
|
|
|
*g.borrow_mut() = Some(Closure::new(move |_timestamp| { |
|
wasm_bindgen_futures::spawn_local(poll_node_graph_evaluation()); |
|
|
|
if !EDITOR_HAS_CRASHED.load(Ordering::SeqCst) { |
|
editor_and_handle(|editor, handle| { |
|
for message in editor.handle_message(InputPreprocessorMessage::CurrentTime { |
|
timestamp: js_sys::Date::now() as u64, |
|
}) { |
|
handle.send_frontend_message_to_js(message); |
|
} |
|
|
|
for message in editor.handle_message(AnimationMessage::IncrementFrameCounter) { |
|
handle.send_frontend_message_to_js(message); |
|
} |
|
|
|
|
|
|
|
for message in editor.handle_message(BroadcastMessage::TriggerEvent(BroadcastEvent::AnimationFrame)) { |
|
handle.send_frontend_message_to_js(message); |
|
} |
|
}); |
|
} |
|
|
|
|
|
request_animation_frame(f.borrow().as_ref().unwrap()); |
|
})); |
|
|
|
request_animation_frame(g.borrow().as_ref().unwrap()); |
|
} |
|
|
|
|
|
{ |
|
let f = std::rc::Rc::new(RefCell::new(None)); |
|
let g = f.clone(); |
|
|
|
*g.borrow_mut() = Some(Closure::new(move || { |
|
auto_save_all_documents(); |
|
|
|
|
|
set_timeout(f.borrow().as_ref().unwrap(), Duration::from_secs(editor::consts::AUTO_SAVE_TIMEOUT_SECONDS)); |
|
})); |
|
|
|
set_timeout(g.borrow().as_ref().unwrap(), Duration::from_secs(editor::consts::AUTO_SAVE_TIMEOUT_SECONDS)); |
|
} |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = errorDialog)] |
|
pub fn error_dialog(&self, title: String, description: String) { |
|
let message = DialogMessage::DisplayDialogError { title, description }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = hasCrashed)] |
|
pub fn has_crashed(&self) -> bool { |
|
EDITOR_HAS_CRASHED.load(Ordering::SeqCst) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = inDevelopmentMode)] |
|
pub fn in_development_mode(&self) -> bool { |
|
cfg!(debug_assertions) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = fileSaveSuffix)] |
|
pub fn file_save_suffix(&self) -> String { |
|
FILE_SAVE_SUFFIX.into() |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = widgetValueUpdate)] |
|
pub fn widget_value_update(&self, layout_target: JsValue, widget_id: u64, value: JsValue) -> Result<(), JsValue> { |
|
let widget_id = WidgetId(widget_id); |
|
match (from_value(layout_target), from_value(value)) { |
|
(Ok(layout_target), Ok(value)) => { |
|
let message = LayoutMessage::WidgetValueUpdate { layout_target, widget_id, value }; |
|
self.dispatch(message); |
|
Ok(()) |
|
} |
|
(target, val) => Err(Error::new(&format!("Could not update UI\nDetails:\nTarget: {target:?}\nValue: {val:?}")).into()), |
|
} |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = widgetValueCommit)] |
|
pub fn widget_value_commit(&self, layout_target: JsValue, widget_id: u64, value: JsValue) -> Result<(), JsValue> { |
|
let widget_id = WidgetId(widget_id); |
|
match (from_value(layout_target), from_value(value)) { |
|
(Ok(layout_target), Ok(value)) => { |
|
let message = LayoutMessage::WidgetValueCommit { layout_target, widget_id, value }; |
|
self.dispatch(message); |
|
Ok(()) |
|
} |
|
(target, val) => Err(Error::new(&format!("Could not commit UI\nDetails:\nTarget: {target:?}\nValue: {val:?}")).into()), |
|
} |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = widgetValueCommitAndUpdate)] |
|
pub fn widget_value_commit_and_update(&self, layout_target: JsValue, widget_id: u64, value: JsValue) -> Result<(), JsValue> { |
|
self.widget_value_commit(layout_target.clone(), widget_id, value.clone())?; |
|
self.widget_value_update(layout_target, widget_id, value)?; |
|
Ok(()) |
|
} |
|
|
|
#[wasm_bindgen(js_name = loadPreferences)] |
|
pub fn load_preferences(&self, preferences: String) { |
|
let message = PreferencesMessage::Load { preferences }; |
|
|
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = selectDocument)] |
|
pub fn select_document(&self, document_id: u64) { |
|
let document_id = DocumentId(document_id); |
|
let message = PortfolioMessage::SelectDocument { document_id }; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = newDocumentDialog)] |
|
pub fn new_document_dialog(&self) { |
|
let message = DialogMessage::RequestNewDocumentDialog; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = openDocument)] |
|
pub fn open_document(&self) { |
|
let message = PortfolioMessage::OpenDocument; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = demoArtworkDialog)] |
|
pub fn demo_artwork_dialog(&self) { |
|
let message = DialogMessage::RequestDemoArtworkDialog; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = openDocumentFile)] |
|
pub fn open_document_file(&self, document_name: String, document_serialized_content: String) { |
|
let message = PortfolioMessage::OpenDocumentFile { |
|
document_name, |
|
document_serialized_content, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = openAutoSavedDocument)] |
|
pub fn open_auto_saved_document(&self, document_id: u64, document_name: String, document_is_saved: bool, document_serialized_content: String, to_front: bool) { |
|
let document_id = DocumentId(document_id); |
|
let message = PortfolioMessage::OpenDocumentFileWithId { |
|
document_id, |
|
document_name, |
|
document_is_auto_saved: true, |
|
document_is_saved, |
|
document_serialized_content, |
|
to_front, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = triggerAutoSave)] |
|
pub fn trigger_auto_save(&self, document_id: u64) { |
|
let document_id = DocumentId(document_id); |
|
let message = PortfolioMessage::AutoSaveDocument { document_id }; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = closeDocumentWithConfirmation)] |
|
pub fn close_document_with_confirmation(&self, document_id: u64) { |
|
let document_id = DocumentId(document_id); |
|
let message = PortfolioMessage::CloseDocumentWithConfirmation { document_id }; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = requestAboutGraphiteDialogWithLocalizedCommitDate)] |
|
pub fn request_about_graphite_dialog_with_localized_commit_date(&self, localized_commit_date: String, localized_commit_year: String) { |
|
let message = DialogMessage::RequestAboutGraphiteDialogWithLocalizedCommitDate { |
|
localized_commit_date, |
|
localized_commit_year, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
|
|
#[wasm_bindgen(js_name = boundsOfViewports)] |
|
pub fn bounds_of_viewports(&self, bounds_of_viewports: &[f64]) { |
|
let chunked: Vec<_> = bounds_of_viewports.chunks(4).map(ViewportBounds::from_slice).collect(); |
|
|
|
let message = InputPreprocessorMessage::BoundsOfViewports { bounds_of_viewports: chunked }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = zoomCanvasToFitAll)] |
|
pub fn zoom_canvas_to_fit_all(&self) { |
|
let message = DocumentMessage::ZoomCanvasToFitAll; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setDevicePixelRatio)] |
|
pub fn set_device_pixel_ratio(&self, ratio: f64) { |
|
let message = PortfolioMessage::SetDevicePixelRatio { ratio }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onMouseMove)] |
|
pub fn on_mouse_move(&self, x: f64, y: f64, mouse_keys: u8, modifiers: u8) { |
|
let editor_mouse_state = EditorMouseState::from_keys_and_editor_position(mouse_keys, (x, y).into()); |
|
|
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
let message = InputPreprocessorMessage::PointerMove { editor_mouse_state, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onWheelScroll)] |
|
pub fn on_wheel_scroll(&self, x: f64, y: f64, mouse_keys: u8, wheel_delta_x: f64, wheel_delta_y: f64, wheel_delta_z: f64, modifiers: u8) { |
|
let mut editor_mouse_state = EditorMouseState::from_keys_and_editor_position(mouse_keys, (x, y).into()); |
|
editor_mouse_state.scroll_delta = ScrollDelta::new(wheel_delta_x, wheel_delta_y, wheel_delta_z); |
|
|
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
let message = InputPreprocessorMessage::WheelScroll { editor_mouse_state, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onMouseDown)] |
|
pub fn on_mouse_down(&self, x: f64, y: f64, mouse_keys: u8, modifiers: u8) { |
|
let editor_mouse_state = EditorMouseState::from_keys_and_editor_position(mouse_keys, (x, y).into()); |
|
|
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
let message = InputPreprocessorMessage::PointerDown { editor_mouse_state, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onMouseUp)] |
|
pub fn on_mouse_up(&self, x: f64, y: f64, mouse_keys: u8, modifiers: u8) { |
|
let editor_mouse_state = EditorMouseState::from_keys_and_editor_position(mouse_keys, (x, y).into()); |
|
|
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
let message = InputPreprocessorMessage::PointerUp { editor_mouse_state, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onDoubleClick)] |
|
pub fn on_double_click(&self, x: f64, y: f64, mouse_keys: u8, modifiers: u8) { |
|
let editor_mouse_state = EditorMouseState::from_keys_and_editor_position(mouse_keys, (x, y).into()); |
|
|
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
let message = InputPreprocessorMessage::DoubleClick { editor_mouse_state, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onKeyDown)] |
|
pub fn on_key_down(&self, name: String, modifiers: u8, key_repeat: bool) { |
|
let key = translate_key(&name); |
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
trace!("Key down {key:?}, name: {name}, modifiers: {modifiers:?}, key repeat: {key_repeat}"); |
|
|
|
let message = InputPreprocessorMessage::KeyDown { key, key_repeat, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onKeyUp)] |
|
pub fn on_key_up(&self, name: String, modifiers: u8, key_repeat: bool) { |
|
let key = translate_key(&name); |
|
let modifier_keys = ModifierKeys::from_bits(modifiers).expect("Invalid modifier keys"); |
|
|
|
trace!("Key up {key:?}, name: {name}, modifiers: {modifier_keys:?}, key repeat: {key_repeat}"); |
|
|
|
let message = InputPreprocessorMessage::KeyUp { key, key_repeat, modifier_keys }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onChangeText)] |
|
pub fn on_change_text(&self, new_text: String, is_left_or_right_click: bool) -> Result<(), JsValue> { |
|
let message = TextToolMessage::TextChange { new_text, is_left_or_right_click }; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = onFontLoad)] |
|
pub fn on_font_load(&self, font_family: String, font_style: String, preview_url: String, data: Vec<u8>) -> Result<(), JsValue> { |
|
let message = PortfolioMessage::FontLoaded { |
|
font_family, |
|
font_style, |
|
preview_url, |
|
data, |
|
}; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = updateBounds)] |
|
pub fn update_bounds(&self, new_text: String) -> Result<(), JsValue> { |
|
let message = TextToolMessage::UpdateBounds { new_text }; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = eyedropperSampleForColorPicker)] |
|
pub fn eyedropper_sample_for_color_picker(&self) -> Result<(), JsValue> { |
|
let message = DialogMessage::RequestComingSoonDialog { issue: Some(832) }; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = updatePrimaryColor)] |
|
pub fn update_primary_color(&self, red: f32, green: f32, blue: f32, alpha: f32) -> Result<(), JsValue> { |
|
let Some(primary_color) = Color::from_rgbaf32(red, green, blue, alpha) else { |
|
return Err(Error::new("Invalid color").into()); |
|
}; |
|
|
|
let message = ToolMessage::SelectWorkingColor { |
|
color: primary_color.to_linear_srgb(), |
|
primary: true, |
|
}; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = updateSecondaryColor)] |
|
pub fn update_secondary_color(&self, red: f32, green: f32, blue: f32, alpha: f32) -> Result<(), JsValue> { |
|
let Some(secondary_color) = Color::from_rgbaf32(red, green, blue, alpha) else { |
|
return Err(Error::new("Invalid color").into()); |
|
}; |
|
|
|
let message = ToolMessage::SelectWorkingColor { |
|
color: secondary_color.to_linear_srgb(), |
|
primary: false, |
|
}; |
|
self.dispatch(message); |
|
|
|
Ok(()) |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = visitUrl)] |
|
pub fn visit_url(&self, url: String) { |
|
let message = FrontendMessage::TriggerVisitLink { url }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = pasteSerializedData)] |
|
pub fn paste_serialized_data(&self, data: String) { |
|
let message = PortfolioMessage::PasteSerializedData { data }; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = clipLayer)] |
|
pub fn clip_layer(&self, id: u64) { |
|
let id = NodeId(id); |
|
let message = DocumentMessage::ClipLayer { id }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = selectLayer)] |
|
pub fn select_layer(&self, id: u64, ctrl: bool, shift: bool) { |
|
let id = NodeId(id); |
|
let message = DocumentMessage::SelectLayer { id, ctrl, shift }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = deselectAllLayers)] |
|
pub fn deselect_all_layers(&self) { |
|
let message = DocumentMessage::DeselectAllLayers; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
|
|
|
|
#[wasm_bindgen(js_name = moveLayerInTree)] |
|
pub fn move_layer_in_tree(&self, insert_parent_id: Option<u64>, insert_index: Option<usize>) { |
|
let insert_parent_id = insert_parent_id.map(NodeId); |
|
let parent = insert_parent_id.map(LayerNodeIdentifier::new_unchecked).unwrap_or_default(); |
|
|
|
let message = DocumentMessage::MoveSelectedLayersTo { |
|
parent, |
|
insert_index: insert_index.unwrap_or_default(), |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setLayerName)] |
|
pub fn set_layer_name(&self, id: u64, name: String) { |
|
let layer = LayerNodeIdentifier::new_unchecked(NodeId(id)); |
|
let message = NodeGraphMessage::SetDisplayName { |
|
node_id: layer.to_node(), |
|
alias: name, |
|
skip_adding_history_step: false, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = panCanvasAbortPrepare)] |
|
pub fn pan_canvas_abort_prepare(&self, x_not_y_axis: bool) { |
|
let message = NavigationMessage::CanvasPanAbortPrepare { x_not_y_axis }; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = panCanvasAbort)] |
|
pub fn pan_canvas_abort(&self, x_not_y_axis: bool) { |
|
let message = NavigationMessage::CanvasPanAbort { x_not_y_axis }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = panCanvas)] |
|
pub fn pan_canvas(&self, delta_x: f64, delta_y: f64) { |
|
let message = NavigationMessage::CanvasPan { delta: (delta_x, delta_y).into() }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = panCanvasByFraction)] |
|
pub fn pan_canvas_by_fraction(&self, delta_x: f64, delta_y: f64) { |
|
let message = NavigationMessage::CanvasPanByViewportFraction { delta: (delta_x, delta_y).into() }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setGridAlignedEdges)] |
|
pub fn set_grid_aligned_edges(&self) { |
|
let message = NodeGraphMessage::SetGridAlignedEdges; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = mergeSelectedNodes)] |
|
pub fn merge_nodes(&self) { |
|
let message = NodeGraphMessage::MergeSelectedNodes; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = createNode)] |
|
pub fn create_node(&self, node_type: String, x: i32, y: i32) { |
|
let id = NodeId::new(); |
|
let message = NodeGraphMessage::CreateNodeFromContextMenu { |
|
node_id: Some(id), |
|
node_type, |
|
xy: Some((x / 24, y / 24)), |
|
add_transaction: true, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = pasteSerializedNodes)] |
|
pub fn paste_serialized_nodes(&self, serialized_nodes: String) { |
|
let message = NodeGraphMessage::PasteNodes { serialized_nodes }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = pasteImage)] |
|
pub fn paste_image( |
|
&self, |
|
name: Option<String>, |
|
image_data: Vec<u8>, |
|
width: u32, |
|
height: u32, |
|
mouse_x: Option<f64>, |
|
mouse_y: Option<f64>, |
|
insert_parent_id: Option<u64>, |
|
insert_index: Option<usize>, |
|
) { |
|
let mouse = mouse_x.and_then(|x| mouse_y.map(|y| (x, y))); |
|
let image = graphene_std::raster::Image::from_image_data(&image_data, width, height); |
|
|
|
let parent_and_insert_index = if let (Some(insert_parent_id), Some(insert_index)) = (insert_parent_id, insert_index) { |
|
let insert_parent_id = NodeId(insert_parent_id); |
|
let parent = LayerNodeIdentifier::new_unchecked(insert_parent_id); |
|
Some((parent, insert_index)) |
|
} else { |
|
None |
|
}; |
|
|
|
let message = PortfolioMessage::PasteImage { |
|
name, |
|
image, |
|
mouse, |
|
parent_and_insert_index, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
#[wasm_bindgen(js_name = pasteSvg)] |
|
pub fn paste_svg(&self, name: Option<String>, svg: String, mouse_x: Option<f64>, mouse_y: Option<f64>, insert_parent_id: Option<u64>, insert_index: Option<usize>) { |
|
let mouse = mouse_x.and_then(|x| mouse_y.map(|y| (x, y))); |
|
|
|
let parent_and_insert_index = if let (Some(insert_parent_id), Some(insert_index)) = (insert_parent_id, insert_index) { |
|
let insert_parent_id = NodeId(insert_parent_id); |
|
let parent = LayerNodeIdentifier::new_unchecked(insert_parent_id); |
|
Some((parent, insert_index)) |
|
} else { |
|
None |
|
}; |
|
|
|
let message = PortfolioMessage::PasteSvg { |
|
name, |
|
svg, |
|
mouse, |
|
parent_and_insert_index, |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = toggleNodeVisibilityLayerPanel)] |
|
pub fn toggle_node_visibility_layer(&self, id: u64) { |
|
let node_id = NodeId(id); |
|
let message = NodeGraphMessage::ToggleVisibility { node_id }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setNodePinned)] |
|
pub fn set_node_pinned(&self, id: u64, pinned: bool) { |
|
self.dispatch(DocumentMessage::SetNodePinned { node_id: NodeId(id), pinned }); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = deleteNode)] |
|
pub fn delete_node(&self, id: u64) { |
|
self.dispatch(DocumentMessage::DeleteNode { node_id: NodeId(id) }); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = toggleLayerLock)] |
|
pub fn toggle_layer_lock(&self, node_id: u64) { |
|
let message = NodeGraphMessage::ToggleLocked { node_id: NodeId(node_id) }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = toggleLayerExpansion)] |
|
pub fn toggle_layer_expansion(&self, id: u64, recursive: bool) { |
|
let id = NodeId(id); |
|
let message = DocumentMessage::ToggleLayerExpansion { id, recursive }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setActivePanel)] |
|
pub fn set_active_panel(&self, panel: String) { |
|
let message = PortfolioMessage::SetActivePanel { panel: panel.into() }; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setToNodeOrLayer)] |
|
pub fn set_to_node_or_layer(&self, id: u64, is_layer: bool) { |
|
self.dispatch(DocumentMessage::SetToNodeOrLayer { node_id: NodeId(id), is_layer }); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setImportName)] |
|
pub fn set_import_name(&self, index: usize, name: String) { |
|
let message = NodeGraphMessage::SetImportExportName { |
|
name, |
|
index: ImportOrExport::Import(index), |
|
}; |
|
self.dispatch(message); |
|
} |
|
|
|
|
|
#[wasm_bindgen(js_name = setExportName)] |
|
pub fn set_export_name(&self, index: usize, name: String) { |
|
let message = NodeGraphMessage::SetImportExportName { |
|
name, |
|
index: ImportOrExport::Export(index), |
|
}; |
|
self.dispatch(message); |
|
} |
|
} |
|
|
|
|
|
|
|
#[wasm_bindgen(js_name = evaluateMathExpression)] |
|
pub fn evaluate_math_expression(expression: &str) -> Option<f64> { |
|
let value = math_parser::evaluate(expression) |
|
.inspect_err(|err| error!("Math parser error on \"{expression}\": {err}")) |
|
.ok()? |
|
.0 |
|
.inspect_err(|err| error!("Math evaluate error on \"{expression}\": {err} ")) |
|
.ok()?; |
|
let Some(real) = value.as_real() else { |
|
error!("{value} was not a real; skipping."); |
|
return None; |
|
}; |
|
Some(real) |
|
} |
|
|
|
|
|
fn request_animation_frame(f: &Closure<dyn FnMut(f64)>) { |
|
web_sys::window() |
|
.expect("No global `window` exists") |
|
.request_animation_frame(f.as_ref().unchecked_ref()) |
|
.expect("Failed to call `requestAnimationFrame`"); |
|
} |
|
|
|
|
|
fn set_timeout(f: &Closure<dyn FnMut()>, delay: Duration) { |
|
let delay = delay.clamp(Duration::ZERO, Duration::from_millis(i32::MAX as u64)).as_millis() as i32; |
|
web_sys::window() |
|
.expect("No global `window` exists") |
|
.set_timeout_with_callback_and_timeout_and_arguments_0(f.as_ref().unchecked_ref(), delay) |
|
.expect("Failed to call `setTimeout`"); |
|
} |
|
|
|
|
|
fn editor<T: Default>(callback: impl FnOnce(&mut editor::application::Editor) -> T) -> T { |
|
EDITOR.with(|editor| { |
|
let mut guard = editor.try_lock(); |
|
let Ok(Some(editor)) = guard.as_deref_mut() else { return T::default() }; |
|
|
|
callback(editor) |
|
}) |
|
} |
|
|
|
|
|
pub(crate) fn editor_and_handle(mut callback: impl FnMut(&mut Editor, &mut EditorHandle)) { |
|
EDITOR_HANDLE.with(|editor_handle| { |
|
editor(|editor| { |
|
let mut guard = editor_handle.try_lock(); |
|
let Ok(Some(editor_handle)) = guard.as_deref_mut() else { |
|
log::error!("Failed to borrow editor handle"); |
|
return; |
|
}; |
|
|
|
|
|
callback(editor, editor_handle); |
|
}) |
|
}); |
|
} |
|
|
|
async fn poll_node_graph_evaluation() { |
|
|
|
if EDITOR_HAS_CRASHED.load(Ordering::SeqCst) { |
|
return; |
|
} |
|
|
|
if !editor::node_graph_executor::run_node_graph().await { |
|
return; |
|
}; |
|
|
|
editor_and_handle(|editor, handle| { |
|
let mut messages = VecDeque::new(); |
|
if let Err(e) = editor.poll_node_graph_evaluation(&mut messages) { |
|
|
|
if e != "No active document" { |
|
error!("Error evaluating node graph:\n{e}"); |
|
} |
|
} |
|
|
|
|
|
if !messages.is_empty() { |
|
crate::NODE_GRAPH_ERROR_DISPLAYED.store(false, Ordering::SeqCst); |
|
} |
|
|
|
|
|
for response in messages.into_iter().flat_map(|message| editor.handle_message(message)) { |
|
handle.send_frontend_message_to_js(response); |
|
} |
|
|
|
|
|
}); |
|
} |
|
|
|
fn auto_save_all_documents() { |
|
|
|
if EDITOR_HAS_CRASHED.load(Ordering::SeqCst) { |
|
return; |
|
} |
|
|
|
editor_and_handle(|editor, handle| { |
|
for message in editor.handle_message(PortfolioMessage::AutoSaveAllDocuments) { |
|
handle.send_frontend_message_to_js(message); |
|
} |
|
}); |
|
} |
|
|