feat: passing multile of the same files in the arguments places a cursor at each position (#12192)

Co-authored-by: Nikita Revenco <154856872+NikitaRevenco@users.noreply.github.com>
Co-authored-by: Michael Davis <mcarsondavis@gmail.com>
This commit is contained in:
Nikita Revenco 2025-01-23 20:04:02 +00:00 committed by GitHub
parent f70923c448
commit 168b11e091
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 45 additions and 17 deletions

1
Cargo.lock generated
View file

@ -1454,6 +1454,7 @@ dependencies = [
"helix-vcs", "helix-vcs",
"helix-view", "helix-view",
"ignore", "ignore",
"indexmap",
"indoc", "indoc",
"libc", "libc",
"log", "log",

View file

@ -619,7 +619,6 @@ impl Selection {
self self
} }
// TODO: consume an iterator or a vec to reduce allocations?
#[must_use] #[must_use]
pub fn new(ranges: SmallVec<[Range; 1]>, primary_index: usize) -> Self { pub fn new(ranges: SmallVec<[Range; 1]>, primary_index: usize) -> Self {
assert!(!ranges.is_empty()); assert!(!ranges.is_empty());
@ -721,6 +720,12 @@ impl IntoIterator for Selection {
} }
} }
impl FromIterator<Range> for Selection {
fn from_iter<T: IntoIterator<Item = Range>>(ranges: T) -> Self {
Self::new(ranges.into_iter().collect(), 0)
}
}
impl From<Range> for Selection { impl From<Range> for Selection {
fn from(range: Range) -> Self { fn from(range: Range) -> Self {
Self { Self {

View file

@ -61,6 +61,7 @@ tokio-stream = "0.1"
futures-util = { version = "0.3", features = ["std", "async-await"], default-features = false } futures-util = { version = "0.3", features = ["std", "async-await"], default-features = false }
arc-swap = { version = "1.7.1" } arc-swap = { version = "1.7.1" }
termini = "1" termini = "1"
indexmap = "2.5"
# Logging # Logging
fern = "0.7" fern = "0.7"

View file

@ -1,6 +1,6 @@
use arc_swap::{access::Map, ArcSwap}; use arc_swap::{access::Map, ArcSwap};
use futures_util::Stream; use futures_util::Stream;
use helix_core::{diagnostic::Severity, pos_at_coords, syntax, Selection}; use helix_core::{diagnostic::Severity, pos_at_coords, syntax, Range, Selection};
use helix_lsp::{ use helix_lsp::{
lsp::{self, notification::Notification}, lsp::{self, notification::Notification},
util::lsp_range_to_range, util::lsp_range_to_range,
@ -210,8 +210,13 @@ impl Application {
// opened last is focused on. // opened last is focused on.
let view_id = editor.tree.focus; let view_id = editor.tree.focus;
let doc = doc_mut!(editor, &doc_id); let doc = doc_mut!(editor, &doc_id);
let pos = Selection::point(pos_at_coords(doc.text().slice(..), pos, true)); let selection = pos
doc.set_selection(view_id, pos); .into_iter()
.map(|coords| {
Range::point(pos_at_coords(doc.text().slice(..), coords, true))
})
.collect();
doc.set_selection(view_id, selection);
} }
} }

View file

@ -1,6 +1,7 @@
use anyhow::Result; use anyhow::Result;
use helix_core::Position; use helix_core::Position;
use helix_view::tree::Layout; use helix_view::tree::Layout;
use indexmap::IndexMap;
use std::path::{Path, PathBuf}; use std::path::{Path, PathBuf};
#[derive(Default)] #[derive(Default)]
@ -16,7 +17,7 @@ pub struct Args {
pub verbosity: u64, pub verbosity: u64,
pub log_file: Option<PathBuf>, pub log_file: Option<PathBuf>,
pub config_file: Option<PathBuf>, pub config_file: Option<PathBuf>,
pub files: Vec<(PathBuf, Position)>, pub files: IndexMap<PathBuf, Vec<Position>>,
pub working_directory: Option<PathBuf>, pub working_directory: Option<PathBuf>,
} }
@ -26,6 +27,18 @@ impl Args {
let mut argv = std::env::args().peekable(); let mut argv = std::env::args().peekable();
let mut line_number = 0; let mut line_number = 0;
let mut insert_file_with_position = |file_with_position: &str| {
let (filename, position) = parse_file(file_with_position);
// Before setting the working directory, resolve all the paths in args.files
let filename = helix_stdx::path::canonicalize(filename);
args.files
.entry(filename)
.and_modify(|positions| positions.push(position))
.or_insert_with(|| vec![position]);
};
argv.next(); // skip the program, we don't care about that argv.next(); // skip the program, we don't care about that
while let Some(arg) = argv.next() { while let Some(arg) = argv.next() {
@ -92,21 +105,25 @@ impl Args {
arg if arg.starts_with('+') => { arg if arg.starts_with('+') => {
match arg[1..].parse::<usize>() { match arg[1..].parse::<usize>() {
Ok(n) => line_number = n.saturating_sub(1), Ok(n) => line_number = n.saturating_sub(1),
_ => args.files.push(parse_file(arg)), _ => insert_file_with_position(arg),
}; };
} }
arg => args.files.push(parse_file(arg)), arg => insert_file_with_position(arg),
} }
} }
// push the remaining args, if any to the files // push the remaining args, if any to the files
for arg in argv { for arg in argv {
args.files.push(parse_file(&arg)); insert_file_with_position(&arg);
} }
if let Some(file) = args.files.first_mut() {
if line_number != 0 { if line_number != 0 {
file.1.row = line_number; if let Some(first_position) = args
.files
.first_mut()
.and_then(|(_, positions)| positions.first_mut())
{
first_position.row = line_number;
} }
} }

View file

@ -40,7 +40,7 @@ fn main() -> Result<()> {
#[tokio::main] #[tokio::main]
async fn main_impl() -> Result<i32> { async fn main_impl() -> Result<i32> {
let mut args = Args::parse_args().context("could not parse arguments")?; let args = Args::parse_args().context("could not parse arguments")?;
helix_loader::initialize_config_file(args.config_file.clone()); helix_loader::initialize_config_file(args.config_file.clone());
helix_loader::initialize_log_file(args.log_file.clone()); helix_loader::initialize_log_file(args.log_file.clone());
@ -114,10 +114,6 @@ FLAGS:
setup_logging(args.verbosity).context("failed to initialize logging")?; setup_logging(args.verbosity).context("failed to initialize logging")?;
// Before setting the working directory, resolve all the paths in args.files
for (path, _) in &mut args.files {
*path = helix_stdx::path::canonicalize(&*path);
}
// NOTE: Set the working directory early so the correct configuration is loaded. Be aware that // NOTE: Set the working directory early so the correct configuration is loaded. Be aware that
// Application::new() depends on this logic so it must be updated if this changes. // Application::new() depends on this logic so it must be updated if this changes.
if let Some(path) = &args.working_directory { if let Some(path) = &args.working_directory {

View file

@ -345,7 +345,10 @@ impl AppBuilder {
path: P, path: P,
pos: Option<helix_core::Position>, pos: Option<helix_core::Position>,
) -> Self { ) -> Self {
self.args.files.push((path.into(), pos.unwrap_or_default())); self.args
.files
.insert(path.into(), vec![pos.unwrap_or_default()]);
self self
} }