Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Copy-pasting nodes #7618

Merged
merged 18 commits into from
Aug 30, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -219,6 +219,10 @@
<kbd>enter</kbd>][7527]
- [Connections to lamdas are displayed correctly][7550]. It is possible to drag
a connection to any expression inside the lambda body.
- [Copying and pasting a single node][7618]. Using the common
<kbd>cmd</kbd>+<kbd>C</kbd> and <kbd>cmd</kbd>+<kbd>V</kbd> shortcuts, it is
now possible to copy a single selected node and paste its code to the graph or
another program.

[5910]: https://github.com/enso-org/enso/pull/5910
[6279]: https://github.com/enso-org/enso/pull/6279
Expand All @@ -245,6 +249,7 @@
[7311]: https://github.com/enso-org/enso/pull/7311
[7527]: https://github.com/enso-org/enso/pull/7527
[7550]: https://github.com/enso-org/enso/pull/7550
[7618]: https://github.com/enso-org/enso/pull/7618

#### EnsoGL (rendering engine)

Expand Down
4 changes: 2 additions & 2 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion app/gui/controller/double-representation/src/node.rs
Original file line number Diff line number Diff line change
Expand Up @@ -458,7 +458,7 @@ impl NodeAst {

/// AST of the node's expression. Typically no external user wants to access it directly. Use
/// [`Self::expression`] instead.
Comment on lines 459 to 460
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Docs accurate?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, it's still not typical to use it, but in this case we need it.

fn whole_expression(&self) -> &Ast {
pub fn whole_expression(&self) -> &Ast {
match self {
NodeAst::Binding { infix, .. } => &infix.rarg,
NodeAst::Expression { ast, .. } => ast,
Expand Down
2 changes: 2 additions & 0 deletions app/gui/docs/product/shortcuts.md
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,8 @@ broken and require further investigation.
| <kbd>cmd</kbd>+<kbd>alt</kbd>+<kbd>r</kbd> | Re-execute the program |
| <kbd>cmd</kbd>+<kbd>shift</kbd>+<kbd>k</kbd> | Switch the execution environment to Design. |
| <kbd>cmd</kbd>+<kbd>shift</kbd>+<kbd>l</kbd> | Switch the execution environment to Live. |
| <kbd>cmd</kbd>+<kbd>c</kbd> | Copy the selected nodes to the clipboard. |
| <kbd>cmd</kbd>+<kbd>v</kbd> | Paste a node from the clipboard at the mouse cursor position. |

#### Navigation

Expand Down
20 changes: 20 additions & 0 deletions app/gui/src/controller/graph.rs
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,10 @@ pub use double_representation::graph::LocationHint;



mod clipboard;



// ==============
// === Errors ===
// ==============
Expand Down Expand Up @@ -929,6 +933,22 @@ impl Handle {
Ok(())
}

/// Copy the node to clipboard. See `clipboard` module documentation for details.
pub fn copy_node(&self, id: ast::Id) -> FallibleResult {
let graph = GraphInfo::from_definition(self.definition()?.item);
let node = graph.locate_node(id)?;
let expression = node.whole_expression().repr();
let metadata = self.module.node_metadata(id).ok();
clipboard::copy_node(expression, metadata)?;
Ok(())
}

/// Paste a node from clipboard at cursor position. See `clipboard` module documentation for
/// details.
pub fn paste_node(&self, cursor_pos: Vector2, on_error: fn(String)) {
clipboard::paste_node(self, cursor_pos, on_error);
}

/// Sets the given's node expression.
#[profile(Debug)]
pub fn set_expression(&self, id: ast::Id, expression_text: impl Str) -> FallibleResult {
Expand Down
172 changes: 172 additions & 0 deletions app/gui/src/controller/graph/clipboard.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
//! Copy-pasting nodes using the clipboard.
//!
//! # Clipboard Content Format
//!
//! We use a JSON-encoded [`ClipboardContent`] structure, marked with our custom [`MIME_TYPE`].
//! This way, we have a separate clipboard format for our application and can extend it in the
//! future.
//! We also support plain text pasting to make it easier to paste the content from other
//! applications, but only if the [`PLAIN_TEXT_PASTING_ENABLED`] is `true`. Allowing pasting plain
//! text can bring unnecessary security risks, like the execution of malicious code immediately
//! after pasting.
//!
//! To copy the node as plain text, the user can enter the editing node, select the node expression,
//! and copy it to the clipboard using the [`ensogl::Text`] functionality.

use crate::prelude::*;

use crate::controller::graph::Handle;
use crate::controller::graph::NewNodeInfo;
use crate::model::module::NodeMetadata;

use ensogl::system::web::clipboard;
use serde::Deserialize;
use serde::Serialize;



// =================
// === Constants ===
// =================

/// We use the `web` prefix to be able to use a custom MIME type. Typically browsers support a
/// restricted set of MIME types in the clipboard.
/// See [Clipboard pickling](https://github.com/w3c/editing/blob/gh-pages/docs/clipboard-pickling/explainer.md).
///
/// `application/enso` is not an officially registered MIME-type (yet), but it is not important for
/// our purposes.
const MIME_TYPE: &str = "web application/enso";
/// Whether to allow pasting nodes from plain text.
const PLAIN_TEXT_PASTING_ENABLED: bool = true;



// ==============
// === Errors ===
// ==============

#[derive(Debug, Clone, PartialEq, failure::Fail)]
#[fail(
display = "`application/enso` MIME-type is used, but clipboard content has incorrect format."
)]
pub struct InvalidFormatError;

/// Clipboard payload.
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
enum ClipboardContent {
/// A single node that was copied from the application.
Node(CopiedNode),
}

/// A single node that was copied from the application.
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
struct CopiedNode {
/// A whole node's expression (without a pattern).
expression: String,
/// Node's metadata.
metadata: Option<NodeMetadata>,
}

/// Copy the node to the clipboard.
pub fn copy_node(expression: String, metadata: Option<NodeMetadata>) -> FallibleResult {
let text_data = Some(expression.clone());
let content = ClipboardContent::Node(CopiedNode { expression, metadata });
let text_repr = serde_json::to_string(&content)?;
clipboard::write(text_repr.as_bytes(), MIME_TYPE.to_string(), text_data);
Ok(())
}


/// Paste the node from the clipboard at a specific position.
///
/// As pasting is an asynchronous operation, we need to provide a callback for handling possible
/// errors.
pub fn paste_node(graph: &Handle, position: Vector2, on_error: fn(String)) {
clipboard::read(
MIME_TYPE.to_string(),
paste_node_from_custom_format(graph, position, on_error),
plain_text_fallback(graph, position, on_error),
);
}

/// A standard callback for pasting node using our custom format.
fn paste_node_from_custom_format(
graph: &Handle,
position: Vector2,
on_error: impl Fn(String) + 'static,
) -> impl Fn(Vec<u8>) + 'static {
let graph = graph.clone_ref();
let closure = move |content| -> FallibleResult {
let _transaction = graph.module.get_or_open_transaction("Paste node");
let string = String::from_utf8(content)?;
if let Ok(content) = serde_json::from_str(&string) {
match content {
ClipboardContent::Node(node) => {
let expression = node.expression;
let metadata = node.metadata;
graph.new_node_at_position(position, expression, metadata)?;
Ok(())
}
}
} else {
Err(InvalidFormatError.into())
}
};
move |content| {
if let Err(err) = closure(content) {
on_error(format!("Failed to paste node. {err}"));
}
}
}

/// An alternative callback for pasting node from plain text. It is used when [`MIME_TYPE`] is not
/// available in the clipboard, and only if [`PLAIN_TEXT_PASTING_ENABLED`]. Otherwise, it is a
/// noop.
fn plain_text_fallback(
graph: &Handle,
position: Vector2,
on_error: impl Fn(String) + 'static,
) -> impl Fn(String) + 'static {
let graph = graph.clone_ref();
let closure = move |text| -> FallibleResult {
if PLAIN_TEXT_PASTING_ENABLED {
let _transaction = graph.module.get_or_open_transaction("Paste node");
let expression = text;
graph.new_node_at_position(position, expression, None)?;
}
Ok(())
};
move |text| {
if let Err(err) = closure(text) {
on_error(format!("Failed to paste node. {err}"));
}
}
}



// ===============
// === Helpers ===
// ===============

impl Handle {
/// Create a new node at the provided position.
fn new_node_at_position(
&self,
position: Vector2,
expression: String,
metadata: Option<NodeMetadata>,
) -> FallibleResult {
let info = NewNodeInfo {
expression,
doc_comment: None,
metadata,
id: None,
location_hint: double_representation::graph::LocationHint::End,
introduce_pattern: true,
};
let ast_id = self.add_node(info)?;
self.set_node_position(ast_id, position)?;
Ok(())
}
}
6 changes: 3 additions & 3 deletions app/gui/src/controller/ide.rs
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@ impl StatusNotificationPublisher {
pub fn publish_event(&self, label: impl Into<String>) {
let label = label.into();
let notification = StatusNotification::Event { label };
executor::global::spawn(self.publisher.publish(notification));
self.publisher.notify(notification);
}

/// Publish a notification about new process (see [`StatusNotification::ProcessStarted`]).
Expand All @@ -69,7 +69,7 @@ impl StatusNotificationPublisher {
let label = label.into();
let handle = Uuid::new_v4();
let notification = StatusNotification::BackgroundTaskStarted { label, handle };
executor::global::spawn(self.publisher.publish(notification));
self.publisher.notify(notification);
handle
}

Expand All @@ -78,7 +78,7 @@ impl StatusNotificationPublisher {
#[profile(Debug)]
pub fn published_background_task_finished(&self, handle: BackgroundTaskHandle) {
let notification = StatusNotification::BackgroundTaskFinished { handle };
executor::global::spawn(self.publisher.publish(notification));
self.publisher.notify(notification);
}

/// The asynchronous stream of published notifications.
Expand Down
23 changes: 22 additions & 1 deletion app/gui/src/presenter/graph.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ use ide_view::graph_editor::component::node as node_view;
use ide_view::graph_editor::component::visualization as visualization_view;
use span_tree::generate::Context as _;
use view::graph_editor::CallWidgetsConfig;
use view::notification::logged as notification;


// ==============
Expand Down Expand Up @@ -291,6 +292,16 @@ impl Model {
Some((node_id, config))
}

fn node_copied(&self, id: ViewNodeId) {
self.log_action(
|| {
let ast_id = self.state.ast_node_id_of_view(id)?;
Some(self.controller.graph().copy_node(ast_id))
},
"copy node",
)
}

/// Node was removed in view.
fn node_removed(&self, id: ViewNodeId) {
self.log_action(
Expand Down Expand Up @@ -433,6 +444,14 @@ impl Model {
}
}

fn paste_node(&self, cursor_pos: Vector2) {
fn on_error(msg: String) {
error!("Error when pasting node. {}", msg);
notification::error(msg, &None);
}
self.controller.graph().paste_node(cursor_pos, on_error);
}

/// Look through all graph's nodes in AST and set position where it is missing.
#[profile(Debug)]
fn initialize_nodes_positions(&self, default_gap_between_nodes: f32) {
Expand Down Expand Up @@ -751,6 +770,7 @@ impl Graph {

// === Changes from the View ===

eval view.node_copied((node_id) model.node_copied(*node_id));
eval view.node_position_set_batched(((node_id, position)) model.node_position_changed(*node_id, *position));
eval view.node_removed((node_id) model.node_removed(*node_id));
eval view.nodes_collapsed(((nodes, _)) model.nodes_collapsed(nodes));
Expand All @@ -765,8 +785,9 @@ impl Graph {
eval_ view.reopen_file_in_language_server (model.reopen_file_in_ls());


// === Dropping Files ===
// === Dropping Files and Pasting Node ===

eval view.request_paste_node((pos) model.paste_node(*pos));
file_upload_requested <- view.file_dropped.gate(&project_view.drop_files_enabled);
eval file_upload_requested (((file,position)) model.file_dropped(file.clone_ref(),*position));
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,8 +50,6 @@
//! [`SpanWidget::PRIORITY_OVER_OVERRIDE`] as `true` in their implementation. In that case, the
//! override will be applied again on their children that use the same span-tree node.



use crate::prelude::*;

use crate::component::node::input::area::NODE_HEIGHT;
Expand Down
22 changes: 21 additions & 1 deletion app/gui/view/graph-editor/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -536,7 +536,9 @@ ensogl::define_endpoints_2! {
/// opposed to e.g. when loading a graph from a file).
start_node_creation_from_port(),


// === Copy-Paste ===
copy_selected_node(),
paste_node(),


/// Remove all selected nodes from the graph.
Expand Down Expand Up @@ -718,6 +720,13 @@ ensogl::define_endpoints_2! {
node_being_edited (Option<NodeId>),
node_editing (bool),


// === Copy-Paste ===

node_copied(NodeId),
// Paste node at position.
request_paste_node(Vector2),

file_dropped (ensogl_drop_manager::File,Vector2<f32>),

connection_made (Connection),
Expand Down Expand Up @@ -3005,6 +3014,17 @@ fn init_remaining_graph_editor_frp(
}


// === Copy-Paste ===

frp::extend! { network
out.node_copied <+ inputs.copy_selected_node.map(f_!(model.nodes.last_selected())).unwrap();
cursor_pos_at_paste <- cursor.scene_position.sample(&inputs.paste_node).map(|v| v.xy());
out.request_paste_node <+ cursor_pos_at_paste.map(
f!([model](pos) new_node_position::at_mouse_aligned_to_close_nodes(&model, *pos))
);
}


// === Set Node Comment ===
frp::extend! { network

Expand Down
Loading