Uncomment more editor code
This commit is contained in:
parent
cdc82d01f7
commit
efa27cf9b8
6 changed files with 1045 additions and 1004 deletions
|
@ -42,6 +42,11 @@ use util::{
|
||||||
// copilot,
|
// copilot,
|
||||||
// [Suggest, NextSuggestion, PreviousSuggestion, Reinstall]
|
// [Suggest, NextSuggestion, PreviousSuggestion, Reinstall]
|
||||||
// );
|
// );
|
||||||
|
//
|
||||||
|
pub struct Suggest;
|
||||||
|
pub struct NextSuggestion;
|
||||||
|
pub struct PreviousSuggestion;
|
||||||
|
pub struct Reinstall;
|
||||||
|
|
||||||
pub fn init(
|
pub fn init(
|
||||||
new_server_id: LanguageServerId,
|
new_server_id: LanguageServerId,
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -261,7 +261,7 @@ pub trait ItemHandle: 'static + Send {
|
||||||
fn act_as_type(&self, type_id: TypeId, cx: &AppContext) -> Option<AnyView>;
|
fn act_as_type(&self, type_id: TypeId, cx: &AppContext) -> Option<AnyView>;
|
||||||
fn to_followable_item_handle(&self, cx: &AppContext) -> Option<Box<dyn FollowableItemHandle>>;
|
fn to_followable_item_handle(&self, cx: &AppContext) -> Option<Box<dyn FollowableItemHandle>>;
|
||||||
fn on_release(
|
fn on_release(
|
||||||
&mut self,
|
&self,
|
||||||
cx: &mut AppContext,
|
cx: &mut AppContext,
|
||||||
callback: Box<dyn FnOnce(&mut AppContext) + Send>,
|
callback: Box<dyn FnOnce(&mut AppContext) + Send>,
|
||||||
) -> gpui2::Subscription;
|
) -> gpui2::Subscription;
|
||||||
|
@ -578,7 +578,7 @@ impl<T: Item> ItemHandle for View<T> {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn on_release(
|
fn on_release(
|
||||||
&mut self,
|
&self,
|
||||||
cx: &mut AppContext,
|
cx: &mut AppContext,
|
||||||
callback: Box<dyn FnOnce(&mut AppContext) + Send>,
|
callback: Box<dyn FnOnce(&mut AppContext) + Send>,
|
||||||
) -> gpui2::Subscription {
|
) -> gpui2::Subscription {
|
||||||
|
|
|
@ -256,6 +256,7 @@ fn main() {
|
||||||
.detach();
|
.detach();
|
||||||
}
|
}
|
||||||
Ok(Some(OpenRequest::JoinChannel { channel_id: _ })) => {
|
Ok(Some(OpenRequest::JoinChannel { channel_id: _ })) => {
|
||||||
|
todo!()
|
||||||
// triggered_authentication = true;
|
// triggered_authentication = true;
|
||||||
// let app_state = app_state.clone();
|
// let app_state = app_state.clone();
|
||||||
// let client = client.clone();
|
// let client = client.clone();
|
||||||
|
@ -267,6 +268,9 @@ fn main() {
|
||||||
// })
|
// })
|
||||||
// .detach_and_log_err(cx)
|
// .detach_and_log_err(cx)
|
||||||
}
|
}
|
||||||
|
Ok(Some(OpenRequest::OpenChannelNotes { channel_id: _ })) => {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
Ok(None) | Err(_) => cx
|
Ok(None) | Err(_) => cx
|
||||||
.spawn({
|
.spawn({
|
||||||
let app_state = app_state.clone();
|
let app_state = app_state.clone();
|
||||||
|
@ -276,28 +280,25 @@ fn main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
let app_state = app_state.clone();
|
let app_state = app_state.clone();
|
||||||
cx.spawn(|cx| {
|
cx.spawn(|cx| async move {
|
||||||
async move {
|
while let Some(request) = open_rx.next().await {
|
||||||
while let Some(request) = open_rx.next().await {
|
match request {
|
||||||
match request {
|
OpenRequest::Paths { paths } => {
|
||||||
OpenRequest::Paths { paths } => {
|
cx.update(|cx| open_paths_and_log_errs(&paths, &app_state, cx))
|
||||||
cx.update(|cx| open_paths_and_log_errs(&paths, &app_state, cx))
|
.ok();
|
||||||
.ok();
|
}
|
||||||
}
|
OpenRequest::CliConnection { connection } => {
|
||||||
OpenRequest::CliConnection { connection } => {
|
let app_state = app_state.clone();
|
||||||
let app_state = app_state.clone();
|
cx.spawn(move |cx| {
|
||||||
cx.spawn(move |cx| {
|
handle_cli_connection(connection, app_state.clone(), cx)
|
||||||
handle_cli_connection(connection, app_state.clone(), cx)
|
})
|
||||||
})
|
.detach();
|
||||||
.detach();
|
}
|
||||||
}
|
OpenRequest::JoinChannel { channel_id: _ } => {
|
||||||
OpenRequest::JoinChannel { channel_id: _ } => {
|
todo!()
|
||||||
// cx
|
}
|
||||||
// .update(|cx| {
|
OpenRequest::OpenChannelNotes { channel_id: _ } => {
|
||||||
// workspace::join_channel(channel_id, app_state.clone(), None, cx)
|
todo!()
|
||||||
// })
|
|
||||||
// .detach()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,15 +1,26 @@
|
||||||
use anyhow::anyhow;
|
use anyhow::{anyhow, Context, Result};
|
||||||
|
use cli::{ipc, IpcHandshake};
|
||||||
use cli::{ipc::IpcSender, CliRequest, CliResponse};
|
use cli::{ipc::IpcSender, CliRequest, CliResponse};
|
||||||
use futures::channel::mpsc;
|
use editor::scroll::autoscroll::Autoscroll;
|
||||||
|
use editor::Editor;
|
||||||
use futures::channel::mpsc::{UnboundedReceiver, UnboundedSender};
|
use futures::channel::mpsc::{UnboundedReceiver, UnboundedSender};
|
||||||
|
use futures::channel::{mpsc, oneshot};
|
||||||
|
use futures::{FutureExt, SinkExt, StreamExt};
|
||||||
|
use gpui::AsyncAppContext;
|
||||||
|
use language::{Bias, Point};
|
||||||
|
use std::collections::HashMap;
|
||||||
use std::ffi::OsStr;
|
use std::ffi::OsStr;
|
||||||
use std::os::unix::prelude::OsStrExt;
|
use std::os::unix::prelude::OsStrExt;
|
||||||
|
use std::path::Path;
|
||||||
use std::sync::atomic::Ordering;
|
use std::sync::atomic::Ordering;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use std::thread;
|
||||||
|
use std::time::Duration;
|
||||||
use std::{path::PathBuf, sync::atomic::AtomicBool};
|
use std::{path::PathBuf, sync::atomic::AtomicBool};
|
||||||
use util::channel::parse_zed_link;
|
use util::channel::parse_zed_link;
|
||||||
|
use util::paths::PathLikeWithPosition;
|
||||||
use util::ResultExt;
|
use util::ResultExt;
|
||||||
|
use workspace::AppState;
|
||||||
use crate::connect_to_cli;
|
|
||||||
|
|
||||||
pub enum OpenRequest {
|
pub enum OpenRequest {
|
||||||
Paths {
|
Paths {
|
||||||
|
@ -21,6 +32,9 @@ pub enum OpenRequest {
|
||||||
JoinChannel {
|
JoinChannel {
|
||||||
channel_id: u64,
|
channel_id: u64,
|
||||||
},
|
},
|
||||||
|
OpenChannelNotes {
|
||||||
|
channel_id: u64,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct OpenListener {
|
pub struct OpenListener {
|
||||||
|
@ -74,7 +88,11 @@ impl OpenListener {
|
||||||
if let Some(slug) = parts.next() {
|
if let Some(slug) = parts.next() {
|
||||||
if let Some(id_str) = slug.split("-").last() {
|
if let Some(id_str) = slug.split("-").last() {
|
||||||
if let Ok(channel_id) = id_str.parse::<u64>() {
|
if let Ok(channel_id) = id_str.parse::<u64>() {
|
||||||
return Some(OpenRequest::JoinChannel { channel_id });
|
if Some("notes") == parts.next() {
|
||||||
|
return Some(OpenRequest::OpenChannelNotes { channel_id });
|
||||||
|
} else {
|
||||||
|
return Some(OpenRequest::JoinChannel { channel_id });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -96,3 +114,191 @@ impl OpenListener {
|
||||||
Some(OpenRequest::Paths { paths })
|
Some(OpenRequest::Paths { paths })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn connect_to_cli(
|
||||||
|
server_name: &str,
|
||||||
|
) -> Result<(mpsc::Receiver<CliRequest>, IpcSender<CliResponse>)> {
|
||||||
|
let handshake_tx = cli::ipc::IpcSender::<IpcHandshake>::connect(server_name.to_string())
|
||||||
|
.context("error connecting to cli")?;
|
||||||
|
let (request_tx, request_rx) = ipc::channel::<CliRequest>()?;
|
||||||
|
let (response_tx, response_rx) = ipc::channel::<CliResponse>()?;
|
||||||
|
|
||||||
|
handshake_tx
|
||||||
|
.send(IpcHandshake {
|
||||||
|
requests: request_tx,
|
||||||
|
responses: response_rx,
|
||||||
|
})
|
||||||
|
.context("error sending ipc handshake")?;
|
||||||
|
|
||||||
|
let (mut async_request_tx, async_request_rx) =
|
||||||
|
futures::channel::mpsc::channel::<CliRequest>(16);
|
||||||
|
thread::spawn(move || {
|
||||||
|
while let Ok(cli_request) = request_rx.recv() {
|
||||||
|
if smol::block_on(async_request_tx.send(cli_request)).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok::<_, anyhow::Error>(())
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok((async_request_rx, response_tx))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn handle_cli_connection(
|
||||||
|
(mut requests, responses): (mpsc::Receiver<CliRequest>, IpcSender<CliResponse>),
|
||||||
|
app_state: Arc<AppState>,
|
||||||
|
mut cx: AsyncAppContext,
|
||||||
|
) {
|
||||||
|
if let Some(request) = requests.next().await {
|
||||||
|
match request {
|
||||||
|
CliRequest::Open { paths, wait } => {
|
||||||
|
let mut caret_positions = HashMap::new();
|
||||||
|
|
||||||
|
let paths = if paths.is_empty() {
|
||||||
|
workspace::last_opened_workspace_paths()
|
||||||
|
.await
|
||||||
|
.map(|location| location.paths().to_vec())
|
||||||
|
.unwrap_or_default()
|
||||||
|
} else {
|
||||||
|
paths
|
||||||
|
.into_iter()
|
||||||
|
.filter_map(|path_with_position_string| {
|
||||||
|
let path_with_position = PathLikeWithPosition::parse_str(
|
||||||
|
&path_with_position_string,
|
||||||
|
|path_str| {
|
||||||
|
Ok::<_, std::convert::Infallible>(
|
||||||
|
Path::new(path_str).to_path_buf(),
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.expect("Infallible");
|
||||||
|
let path = path_with_position.path_like;
|
||||||
|
if let Some(row) = path_with_position.row {
|
||||||
|
if path.is_file() {
|
||||||
|
let row = row.saturating_sub(1);
|
||||||
|
let col =
|
||||||
|
path_with_position.column.unwrap_or(0).saturating_sub(1);
|
||||||
|
caret_positions.insert(path.clone(), Point::new(row, col));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(path)
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut errored = false;
|
||||||
|
|
||||||
|
match cx.update(|cx| workspace::open_paths(&paths, &app_state, None, cx)) {
|
||||||
|
Ok(task) => match task.await {
|
||||||
|
Ok((workspace, items)) => {
|
||||||
|
let mut item_release_futures = Vec::new();
|
||||||
|
|
||||||
|
for (item, path) in items.into_iter().zip(&paths) {
|
||||||
|
match item {
|
||||||
|
Some(Ok(item)) => {
|
||||||
|
if let Some(point) = caret_positions.remove(path) {
|
||||||
|
if let Some(active_editor) = item.downcast::<Editor>() {
|
||||||
|
workspace
|
||||||
|
.update(&mut cx, |_, cx| {
|
||||||
|
active_editor.update(cx, |editor, cx| {
|
||||||
|
let snapshot = editor
|
||||||
|
.snapshot(cx)
|
||||||
|
.display_snapshot;
|
||||||
|
let point = snapshot
|
||||||
|
.buffer_snapshot
|
||||||
|
.clip_point(point, Bias::Left);
|
||||||
|
editor.change_selections(
|
||||||
|
Some(Autoscroll::center()),
|
||||||
|
cx,
|
||||||
|
|s| s.select_ranges([point..point]),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.log_err();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cx.update(|cx| {
|
||||||
|
let released = oneshot::channel();
|
||||||
|
item.on_release(
|
||||||
|
cx,
|
||||||
|
Box::new(move |_| {
|
||||||
|
let _ = released.0.send(());
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
.detach();
|
||||||
|
item_release_futures.push(released.1);
|
||||||
|
})
|
||||||
|
.log_err();
|
||||||
|
}
|
||||||
|
Some(Err(err)) => {
|
||||||
|
responses
|
||||||
|
.send(CliResponse::Stderr {
|
||||||
|
message: format!(
|
||||||
|
"error opening {:?}: {}",
|
||||||
|
path, err
|
||||||
|
),
|
||||||
|
})
|
||||||
|
.log_err();
|
||||||
|
errored = true;
|
||||||
|
}
|
||||||
|
None => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if wait {
|
||||||
|
let background = cx.background_executor().clone();
|
||||||
|
let wait = async move {
|
||||||
|
if paths.is_empty() {
|
||||||
|
let (done_tx, done_rx) = oneshot::channel();
|
||||||
|
let _subscription =
|
||||||
|
workspace.update(&mut cx, |workspace, cx| {
|
||||||
|
cx.on_release(move |_, _| {
|
||||||
|
let _ = done_tx.send(());
|
||||||
|
})
|
||||||
|
});
|
||||||
|
let _ = done_rx.await;
|
||||||
|
} else {
|
||||||
|
let _ = futures::future::try_join_all(item_release_futures)
|
||||||
|
.await;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
.fuse();
|
||||||
|
futures::pin_mut!(wait);
|
||||||
|
|
||||||
|
loop {
|
||||||
|
// Repeatedly check if CLI is still open to avoid wasting resources
|
||||||
|
// waiting for files or workspaces to close.
|
||||||
|
let mut timer = background.timer(Duration::from_secs(1)).fuse();
|
||||||
|
futures::select_biased! {
|
||||||
|
_ = wait => break,
|
||||||
|
_ = timer => {
|
||||||
|
if responses.send(CliResponse::Ping).is_err() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(error) => {
|
||||||
|
errored = true;
|
||||||
|
responses
|
||||||
|
.send(CliResponse::Stderr {
|
||||||
|
message: format!("error opening {:?}: {}", paths, error),
|
||||||
|
})
|
||||||
|
.log_err();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Err(_) => errored = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
responses
|
||||||
|
.send(CliResponse::Exit {
|
||||||
|
status: i32::from(errored),
|
||||||
|
})
|
||||||
|
.log_err();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -7,218 +7,18 @@ mod only_instance;
|
||||||
mod open_listener;
|
mod open_listener;
|
||||||
|
|
||||||
pub use assets::*;
|
pub use assets::*;
|
||||||
use collections::HashMap;
|
|
||||||
use gpui::{
|
use gpui::{
|
||||||
point, px, AppContext, AsyncAppContext, AsyncWindowContext, Point, Task, TitlebarOptions,
|
point, px, AppContext, AsyncWindowContext, Task, TitlebarOptions, WeakView, WindowBounds,
|
||||||
WeakView, WindowBounds, WindowKind, WindowOptions,
|
WindowKind, WindowOptions,
|
||||||
};
|
};
|
||||||
pub use only_instance::*;
|
pub use only_instance::*;
|
||||||
pub use open_listener::*;
|
pub use open_listener::*;
|
||||||
|
|
||||||
use anyhow::{Context, Result};
|
use anyhow::Result;
|
||||||
use cli::{
|
use std::sync::Arc;
|
||||||
ipc::{self, IpcSender},
|
|
||||||
CliRequest, CliResponse, IpcHandshake,
|
|
||||||
};
|
|
||||||
use futures::{
|
|
||||||
channel::{mpsc, oneshot},
|
|
||||||
FutureExt, SinkExt, StreamExt,
|
|
||||||
};
|
|
||||||
use std::{path::Path, sync::Arc, thread, time::Duration};
|
|
||||||
use util::{paths::PathLikeWithPosition, ResultExt};
|
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
use workspace::{AppState, Workspace};
|
use workspace::{AppState, Workspace};
|
||||||
|
|
||||||
pub fn connect_to_cli(
|
|
||||||
server_name: &str,
|
|
||||||
) -> Result<(mpsc::Receiver<CliRequest>, IpcSender<CliResponse>)> {
|
|
||||||
let handshake_tx = cli::ipc::IpcSender::<IpcHandshake>::connect(server_name.to_string())
|
|
||||||
.context("error connecting to cli")?;
|
|
||||||
let (request_tx, request_rx) = ipc::channel::<CliRequest>()?;
|
|
||||||
let (response_tx, response_rx) = ipc::channel::<CliResponse>()?;
|
|
||||||
|
|
||||||
handshake_tx
|
|
||||||
.send(IpcHandshake {
|
|
||||||
requests: request_tx,
|
|
||||||
responses: response_rx,
|
|
||||||
})
|
|
||||||
.context("error sending ipc handshake")?;
|
|
||||||
|
|
||||||
let (mut async_request_tx, async_request_rx) =
|
|
||||||
futures::channel::mpsc::channel::<CliRequest>(16);
|
|
||||||
thread::spawn(move || {
|
|
||||||
while let Ok(cli_request) = request_rx.recv() {
|
|
||||||
if smol::block_on(async_request_tx.send(cli_request)).is_err() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok::<_, anyhow::Error>(())
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok((async_request_rx, response_tx))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn handle_cli_connection(
|
|
||||||
(mut requests, responses): (mpsc::Receiver<CliRequest>, IpcSender<CliResponse>),
|
|
||||||
app_state: Arc<AppState>,
|
|
||||||
mut cx: AsyncAppContext,
|
|
||||||
) {
|
|
||||||
if let Some(request) = requests.next().await {
|
|
||||||
match request {
|
|
||||||
CliRequest::Open { paths, wait } => {
|
|
||||||
let mut caret_positions = HashMap::default();
|
|
||||||
|
|
||||||
let paths = if paths.is_empty() {
|
|
||||||
todo!()
|
|
||||||
// workspace::last_opened_workspace_paths()
|
|
||||||
// .await
|
|
||||||
// .map(|location| location.paths().to_vec())
|
|
||||||
// .unwrap_or_default()
|
|
||||||
} else {
|
|
||||||
paths
|
|
||||||
.into_iter()
|
|
||||||
.filter_map(|path_with_position_string| {
|
|
||||||
let path_with_position = PathLikeWithPosition::parse_str(
|
|
||||||
&path_with_position_string,
|
|
||||||
|path_str| {
|
|
||||||
Ok::<_, std::convert::Infallible>(
|
|
||||||
Path::new(path_str).to_path_buf(),
|
|
||||||
)
|
|
||||||
},
|
|
||||||
)
|
|
||||||
.expect("Infallible");
|
|
||||||
let path = path_with_position.path_like;
|
|
||||||
if let Some(row) = path_with_position.row {
|
|
||||||
if path.is_file() {
|
|
||||||
let row = row.saturating_sub(1);
|
|
||||||
let col =
|
|
||||||
path_with_position.column.unwrap_or(0).saturating_sub(1);
|
|
||||||
caret_positions.insert(path.clone(), Point::new(row, col));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(path)
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>()
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut errored = false;
|
|
||||||
|
|
||||||
if let Some(open_paths_task) = cx
|
|
||||||
.update(|cx| workspace::open_paths(&paths, &app_state, None, cx))
|
|
||||||
.log_err()
|
|
||||||
{
|
|
||||||
match open_paths_task.await {
|
|
||||||
Ok((workspace, items)) => {
|
|
||||||
let mut item_release_futures = Vec::new();
|
|
||||||
|
|
||||||
for (item, path) in items.into_iter().zip(&paths) {
|
|
||||||
match item {
|
|
||||||
Some(Ok(mut item)) => {
|
|
||||||
if let Some(point) = caret_positions.remove(path) {
|
|
||||||
todo!()
|
|
||||||
// if let Some(active_editor) = item.downcast::<Editor>() {
|
|
||||||
// active_editor
|
|
||||||
// .downgrade()
|
|
||||||
// .update(&mut cx, |editor, cx| {
|
|
||||||
// let snapshot =
|
|
||||||
// editor.snapshot(cx).display_snapshot;
|
|
||||||
// let point = snapshot
|
|
||||||
// .buffer_snapshot
|
|
||||||
// .clip_point(point, Bias::Left);
|
|
||||||
// editor.change_selections(
|
|
||||||
// Some(Autoscroll::center()),
|
|
||||||
// cx,
|
|
||||||
// |s| s.select_ranges([point..point]),
|
|
||||||
// );
|
|
||||||
// })
|
|
||||||
// .log_err();
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
let released = oneshot::channel();
|
|
||||||
cx.update(move |cx| {
|
|
||||||
item.on_release(
|
|
||||||
cx,
|
|
||||||
Box::new(move |_| {
|
|
||||||
let _ = released.0.send(());
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.detach();
|
|
||||||
})
|
|
||||||
.ok();
|
|
||||||
item_release_futures.push(released.1);
|
|
||||||
}
|
|
||||||
Some(Err(err)) => {
|
|
||||||
responses
|
|
||||||
.send(CliResponse::Stderr {
|
|
||||||
message: format!(
|
|
||||||
"error opening {:?}: {}",
|
|
||||||
path, err
|
|
||||||
),
|
|
||||||
})
|
|
||||||
.log_err();
|
|
||||||
errored = true;
|
|
||||||
}
|
|
||||||
None => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if wait {
|
|
||||||
let executor = cx.background_executor().clone();
|
|
||||||
let wait = async move {
|
|
||||||
if paths.is_empty() {
|
|
||||||
let (done_tx, done_rx) = oneshot::channel();
|
|
||||||
let _subscription =
|
|
||||||
workspace.update(&mut cx, move |_, cx| {
|
|
||||||
cx.on_release(|_, _| {
|
|
||||||
let _ = done_tx.send(());
|
|
||||||
})
|
|
||||||
});
|
|
||||||
let _ = done_rx.await;
|
|
||||||
} else {
|
|
||||||
let _ = futures::future::try_join_all(item_release_futures)
|
|
||||||
.await;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
.fuse();
|
|
||||||
futures::pin_mut!(wait);
|
|
||||||
|
|
||||||
loop {
|
|
||||||
// Repeatedly check if CLI is still open to avoid wasting resources
|
|
||||||
// waiting for files or workspaces to close.
|
|
||||||
let mut timer = executor.timer(Duration::from_secs(1)).fuse();
|
|
||||||
futures::select_biased! {
|
|
||||||
_ = wait => break,
|
|
||||||
_ = timer => {
|
|
||||||
if responses.send(CliResponse::Ping).is_err() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(error) => {
|
|
||||||
errored = true;
|
|
||||||
responses
|
|
||||||
.send(CliResponse::Stderr {
|
|
||||||
message: format!("error opening {:?}: {}", paths, error),
|
|
||||||
})
|
|
||||||
.log_err();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
responses
|
|
||||||
.send(CliResponse::Exit {
|
|
||||||
status: i32::from(errored),
|
|
||||||
})
|
|
||||||
.log_err();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn build_window_options(
|
pub fn build_window_options(
|
||||||
bounds: Option<WindowBounds>,
|
bounds: Option<WindowBounds>,
|
||||||
display_uuid: Option<Uuid>,
|
display_uuid: Option<Uuid>,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue