From 59b94f04603b80f3b23dbb21e83b6f28328b50d3 Mon Sep 17 00:00:00 2001 From: Martin von Zweigbergk Date: Sun, 11 Feb 2024 21:56:52 -0800 Subject: [PATCH] next/prev: make --edit implied when already on non-head commit Users who edit non-head commits usually expect `jj next/prev` to continue to edit the next/previous commit, so let's make that the default behavior. This should not confuse users who don't edit non-head commits since they will simply not be in this state. My main concern is that doing `jj next; jj prev` will now usually take you back to the previous commit, but not if you started on the parent of a head commit. --- CHANGELOG.md | 3 +++ cli/src/commands/next.rs | 10 +++++++++- cli/src/commands/prev.rs | 12 ++++++++++-- cli/tests/cli-reference@.md.snap | 8 +++++++- cli/tests/test_next_prev_commands.rs | 16 +++++++++++++++- 5 files changed, 44 insertions(+), 5 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 003b3ccccf9..7bd77fd025e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -23,6 +23,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 * `jj git fetch` now accepts `-b` as a shorthand for `--branch`, making it more consistent with other commands that accept a branch +* `jj next/prev` now infer `--edit` when you're already editing a non-head + commit (a commit with children). + ### Fixed bugs * On Windows, symlinks in the repo are now materialized as regular files in the diff --git a/cli/src/commands/next.rs b/cli/src/commands/next.rs index 25d623a196e..ead6ecbf945 100644 --- a/cli/src/commands/next.rs +++ b/cli/src/commands/next.rs @@ -48,6 +48,9 @@ use crate::ui::Ui; /// B => @ /// | | /// @ A +/// +/// If your working-copy commit already has visible children, then `--edit` is +/// implied. /// ``` #[derive(clap::Args, Clone, Debug)] #[command(verbatim_doc_comment)] @@ -100,11 +103,16 @@ pub(crate) fn cmd_next( args: &NextArgs, ) -> Result<(), CommandError> { let mut workspace_command = command.workspace_helper(ui)?; - let edit = args.edit; let amount = args.amount; let current_wc_id = workspace_command .get_wc_commit_id() .ok_or_else(|| user_error("This command requires a working copy"))?; + let edit = args.edit + || !workspace_command + .repo() + .view() + .heads() + .contains(current_wc_id); let current_wc = workspace_command.repo().store().get_commit(current_wc_id)?; // If we're editing, start at the working-copy commit. // Otherwise start from our direct parent. diff --git a/cli/src/commands/prev.rs b/cli/src/commands/prev.rs index b39409bf499..2da2fadf2e6 100644 --- a/cli/src/commands/prev.rs +++ b/cli/src/commands/prev.rs @@ -29,7 +29,7 @@ use crate::ui::Ui; /// D @ D /// |/ | /// A => A @ -/// | | / +/// | |/ /// B B /// ``` /// @@ -44,6 +44,9 @@ use crate::ui::Ui; /// B B /// | | /// A A +/// +/// If your working-copy commit already has visible children, then `--edit` is +/// implied. /// ``` // TODO(#2126): Handle multiple parents, e.g merges. #[derive(clap::Args, Clone, Debug)] @@ -63,11 +66,16 @@ pub(crate) fn cmd_prev( args: &PrevArgs, ) -> Result<(), CommandError> { let mut workspace_command = command.workspace_helper(ui)?; - let edit = args.edit; let amount = args.amount; let current_wc_id = workspace_command .get_wc_commit_id() .ok_or_else(|| user_error("This command requires a working copy"))?; + let edit = args.edit + || !workspace_command + .repo() + .view() + .heads() + .contains(current_wc_id); let current_wc = workspace_command.repo().store().get_commit(current_wc_id)?; let start_id = if edit { current_wc_id diff --git a/cli/tests/cli-reference@.md.snap b/cli/tests/cli-reference@.md.snap index c78cb8916e5..84c69eaceb2 100644 --- a/cli/tests/cli-reference@.md.snap +++ b/cli/tests/cli-reference@.md.snap @@ -1145,6 +1145,9 @@ C C B => @ | | @ A + +If your working-copy commit is already a non-head commit (i.e. if it has +visible children), then `--edit` is implied. ``` **Usage:** `jj next [OPTIONS] [AMOUNT]` @@ -1330,7 +1333,7 @@ The command moves you to the parent in a linear fashion. D @ D |/ | A => A @ -| | / +| |/ B B ``` @@ -1345,6 +1348,9 @@ C => @ B B | | A A + +If your working-copy commit is already a non-head commit (i.e. if it has +visible children), then `--edit` is implied. ``` **Usage:** `jj prev [OPTIONS] [AMOUNT]` diff --git a/cli/tests/test_next_prev_commands.rs b/cli/tests/test_next_prev_commands.rs index dbf1ad7a8ad..c42c8829538 100644 --- a/cli/tests/test_next_prev_commands.rs +++ b/cli/tests/test_next_prev_commands.rs @@ -284,6 +284,13 @@ fn test_prev_editing() { Working copy now at: zsuskuln 009f88bf (empty) fourth Parent commit : kkmpptxz 3fa8931e (empty) third "###); + // --edit is implied when already editing a non-head commit + let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["prev"]); + insta::assert_snapshot!(stdout, @""); + insta::assert_snapshot!(stderr, @r###" + Working copy now at: yqosqzyt d2edc95b (empty) (no description set) + Parent commit : rlvkpnrz 5c52832c (empty) second + "###); } #[test] @@ -296,10 +303,17 @@ fn test_next_editing() { test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "second"]); test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "third"]); test_env.jj_cmd_ok(&repo_path, &["commit", "-m", "fourth"]); - test_env.jj_cmd_ok(&repo_path, &["edit", "@--"]); + test_env.jj_cmd_ok(&repo_path, &["edit", "@---"]); let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["next", "--edit"]); insta::assert_snapshot!(stdout, @""); insta::assert_snapshot!(stderr, @r###" + Working copy now at: kkmpptxz 3fa8931e (empty) third + Parent commit : rlvkpnrz 5c52832c (empty) second + "###); + // --edit is implied when already editing a non-head commit + let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["next"]); + insta::assert_snapshot!(stdout, @""); + insta::assert_snapshot!(stderr, @r###" Working copy now at: zsuskuln 009f88bf (empty) fourth Parent commit : kkmpptxz 3fa8931e (empty) third "###);