repl: Process display IDs for updatable displays (#15738)

Release Notes:

- Added `update_display_data` support for REPL.


https://github.com/user-attachments/assets/d618e457-e314-482e-954a-6384f185629a
This commit is contained in:
Kyle Kelley 2024-08-03 08:15:36 -07:00 committed by GitHub
parent 68446d2ed1
commit 36b61a8b87
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 105 additions and 34 deletions

View file

@ -268,7 +268,28 @@ impl ErrorView {
} }
} }
pub enum OutputType { pub struct Output {
content: OutputContent,
display_id: Option<String>,
}
impl Output {
pub fn new(data: &MimeBundle, display_id: Option<String>, cx: &mut WindowContext) -> Self {
Self {
content: OutputContent::new(data, cx),
display_id,
}
}
pub fn from(content: OutputContent) -> Self {
Self {
content,
display_id: None,
}
}
}
pub enum OutputContent {
Plain(TerminalOutput), Plain(TerminalOutput),
Stream(TerminalOutput), Stream(TerminalOutput),
Image(ImageView), Image(ImageView),
@ -278,24 +299,24 @@ pub enum OutputType {
ClearOutputWaitMarker, ClearOutputWaitMarker,
} }
impl std::fmt::Debug for OutputType { impl std::fmt::Debug for OutputContent {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self { match self {
OutputType::Plain(_) => f.debug_struct("OutputType(Plain)"), OutputContent::Plain(_) => f.debug_struct("OutputContent(Plain)"),
OutputType::Stream(_) => f.debug_struct("OutputType(Stream)"), OutputContent::Stream(_) => f.debug_struct("OutputContent(Stream)"),
OutputType::Image(_) => f.debug_struct("OutputType(Image)"), OutputContent::Image(_) => f.debug_struct("OutputContent(Image)"),
OutputType::ErrorOutput(_) => f.debug_struct("OutputType(ErrorOutput)"), OutputContent::ErrorOutput(_) => f.debug_struct("OutputContent(ErrorOutput)"),
OutputType::Message(_) => f.debug_struct("OutputType(Message)"), OutputContent::Message(_) => f.debug_struct("OutputContent(Message)"),
OutputType::Table(_) => f.debug_struct("OutputType(Table)"), OutputContent::Table(_) => f.debug_struct("OutputContent(Table)"),
OutputType::ClearOutputWaitMarker => { OutputContent::ClearOutputWaitMarker => {
f.debug_struct("OutputType(ClearOutputWaitMarker)") f.debug_struct("OutputContent(ClearOutputWaitMarker)")
} }
} }
.finish() .finish()
} }
} }
impl OutputType { impl OutputContent {
fn render(&self, cx: &ViewContext<ExecutionView>) -> Option<AnyElement> { fn render(&self, cx: &ViewContext<ExecutionView>) -> Option<AnyElement> {
let el = match self { let el = match self {
// Note: in typical frontends we would show the execute_result.execution_count // Note: in typical frontends we would show the execute_result.execution_count
@ -315,15 +336,17 @@ impl OutputType {
pub fn new(data: &MimeBundle, cx: &mut WindowContext) -> Self { pub fn new(data: &MimeBundle, cx: &mut WindowContext) -> Self {
match data.richest(rank_mime_type) { match data.richest(rank_mime_type) {
Some(MimeType::Plain(text)) => OutputType::Plain(TerminalOutput::from(text, cx)), Some(MimeType::Plain(text)) => OutputContent::Plain(TerminalOutput::from(text, cx)),
Some(MimeType::Markdown(text)) => OutputType::Plain(TerminalOutput::from(text, cx)), Some(MimeType::Markdown(text)) => OutputContent::Plain(TerminalOutput::from(text, cx)),
Some(MimeType::Png(data)) | Some(MimeType::Jpeg(data)) => match ImageView::from(data) { Some(MimeType::Png(data)) | Some(MimeType::Jpeg(data)) => match ImageView::from(data) {
Ok(view) => OutputType::Image(view), Ok(view) => OutputContent::Image(view),
Err(error) => OutputType::Message(format!("Failed to load image: {}", error)), Err(error) => OutputContent::Message(format!("Failed to load image: {}", error)),
}, },
Some(MimeType::DataTable(data)) => OutputType::Table(TableView::new(data.clone(), cx)), Some(MimeType::DataTable(data)) => {
OutputContent::Table(TableView::new(data.clone(), cx))
}
// Any other media types are not supported // Any other media types are not supported
_ => OutputType::Message("Unsupported media type".to_string()), _ => OutputContent::Message("Unsupported media type".to_string()),
} }
} }
} }
@ -342,7 +365,7 @@ pub enum ExecutionStatus {
} }
pub struct ExecutionView { pub struct ExecutionView {
pub outputs: Vec<OutputType>, pub outputs: Vec<Output>,
pub status: ExecutionStatus, pub status: ExecutionStatus,
} }
@ -356,13 +379,19 @@ impl ExecutionView {
/// Accept a Jupyter message belonging to this execution /// Accept a Jupyter message belonging to this execution
pub fn push_message(&mut self, message: &JupyterMessageContent, cx: &mut ViewContext<Self>) { pub fn push_message(&mut self, message: &JupyterMessageContent, cx: &mut ViewContext<Self>) {
let output: OutputType = match message { let output: Output = match message {
JupyterMessageContent::ExecuteResult(result) => OutputType::new(&result.data, cx), JupyterMessageContent::ExecuteResult(result) => Output::new(
JupyterMessageContent::DisplayData(result) => OutputType::new(&result.data, cx), &result.data,
result.transient.as_ref().and_then(|t| t.display_id.clone()),
cx,
),
JupyterMessageContent::DisplayData(result) => {
Output::new(&result.data, result.transient.display_id.clone(), cx)
}
JupyterMessageContent::StreamContent(result) => { JupyterMessageContent::StreamContent(result) => {
// Previous stream data will combine together, handling colors, carriage returns, etc // Previous stream data will combine together, handling colors, carriage returns, etc
if let Some(new_terminal) = self.apply_terminal_text(&result.text, cx) { if let Some(new_terminal) = self.apply_terminal_text(&result.text, cx) {
new_terminal Output::from(new_terminal)
} else { } else {
return; return;
} }
@ -371,11 +400,11 @@ impl ExecutionView {
let mut terminal = TerminalOutput::new(cx); let mut terminal = TerminalOutput::new(cx);
terminal.append_text(&result.traceback.join("\n")); terminal.append_text(&result.traceback.join("\n"));
OutputType::ErrorOutput(ErrorView { Output::from(OutputContent::ErrorOutput(ErrorView {
ename: result.ename.clone(), ename: result.ename.clone(),
evalue: result.evalue.clone(), evalue: result.evalue.clone(),
traceback: terminal, traceback: terminal,
}) }))
} }
JupyterMessageContent::ExecuteReply(reply) => { JupyterMessageContent::ExecuteReply(reply) => {
for payload in reply.payload.iter() { for payload in reply.payload.iter() {
@ -383,7 +412,7 @@ impl ExecutionView {
// Pager data comes in via `?` at the end of a statement in Python, used for showing documentation. // Pager data comes in via `?` at the end of a statement in Python, used for showing documentation.
// Some UI will show this as a popup. For ease of implementation, it's included as an output here. // Some UI will show this as a popup. For ease of implementation, it's included as an output here.
runtimelib::Payload::Page { data, .. } => { runtimelib::Payload::Page { data, .. } => {
let output = OutputType::new(data, cx); let output = Output::new(data, None, cx);
self.outputs.push(output); self.outputs.push(output);
} }
@ -416,7 +445,7 @@ impl ExecutionView {
} }
// Create a marker to clear the output after we get in a new output // Create a marker to clear the output after we get in a new output
OutputType::ClearOutputWaitMarker Output::from(OutputContent::ClearOutputWaitMarker)
} }
JupyterMessageContent::Status(status) => { JupyterMessageContent::Status(status) => {
match status.execution_state { match status.execution_state {
@ -434,8 +463,10 @@ impl ExecutionView {
}; };
// Check for a clear output marker as the previous output, so we can clear it out // Check for a clear output marker as the previous output, so we can clear it out
if let Some(OutputType::ClearOutputWaitMarker) = self.outputs.last() { if let Some(output) = self.outputs.last() {
self.outputs.clear(); if let OutputContent::ClearOutputWaitMarker = output.content {
self.outputs.clear();
}
} }
self.outputs.push(output); self.outputs.push(output);
@ -443,21 +474,43 @@ impl ExecutionView {
cx.notify(); cx.notify();
} }
pub fn update_display_data(
&mut self,
data: &MimeBundle,
display_id: &str,
cx: &mut ViewContext<Self>,
) {
let mut any = false;
self.outputs.iter_mut().for_each(|output| {
if let Some(other_display_id) = output.display_id.as_ref() {
if other_display_id == display_id {
output.content = OutputContent::new(data, cx);
any = true;
}
}
});
if any {
cx.notify();
}
}
fn apply_terminal_text( fn apply_terminal_text(
&mut self, &mut self,
text: &str, text: &str,
cx: &mut ViewContext<Self>, cx: &mut ViewContext<Self>,
) -> Option<OutputType> { ) -> Option<OutputContent> {
if let Some(last_output) = self.outputs.last_mut() { if let Some(last_output) = self.outputs.last_mut() {
match last_output { match &mut last_output.content {
OutputType::Stream(last_stream) => { OutputContent::Stream(last_stream) => {
last_stream.append_text(text); last_stream.append_text(text);
// Don't need to add a new output, we already have a terminal output // Don't need to add a new output, we already have a terminal output
cx.notify(); cx.notify();
return None; return None;
} }
// Edge case note: a clear output marker // Edge case note: a clear output marker
OutputType::ClearOutputWaitMarker => { OutputContent::ClearOutputWaitMarker => {
// Edge case note: a clear output marker is handled by the caller // Edge case note: a clear output marker is handled by the caller
// since we will return a new output at the end here as a new terminal output // since we will return a new output at the end here as a new terminal output
} }
@ -469,7 +522,7 @@ impl ExecutionView {
let mut new_terminal = TerminalOutput::new(cx); let mut new_terminal = TerminalOutput::new(cx);
new_terminal.append_text(text); new_terminal.append_text(text);
Some(OutputType::Stream(new_terminal)) Some(OutputContent::Stream(new_terminal))
} }
} }
@ -523,7 +576,11 @@ impl Render for ExecutionView {
div() div()
.w_full() .w_full()
.children(self.outputs.iter().filter_map(|output| output.render(cx))) .children(
self.outputs
.iter()
.filter_map(|output| output.content.render(cx)),
)
.children(match self.status { .children(match self.status {
ExecutionStatus::Executing => vec![status], ExecutionStatus::Executing => vec![status],
ExecutionStatus::Queued => vec![status], ExecutionStatus::Queued => vec![status],

View file

@ -549,6 +549,20 @@ impl Session {
self.kernel.set_kernel_info(&reply); self.kernel.set_kernel_info(&reply);
cx.notify(); cx.notify();
} }
JupyterMessageContent::UpdateDisplayData(update) => {
let display_id = if let Some(display_id) = update.transient.display_id.clone() {
display_id
} else {
return;
};
self.blocks.iter_mut().for_each(|(_, block)| {
block.execution_view.update(cx, |execution_view, cx| {
execution_view.update_display_data(&update.data, &display_id, cx);
});
});
return;
}
_ => {} _ => {}
} }