@@ -11,10 +11,33 @@ enum prompt_mode_type {
1111 PROMPT_MODE_CHANGE = 0 , PROMPT_DELETION , PROMPT_HUNK
1212};
1313
14- static const char * prompt_mode [] = {
15- N_ ("Stage mode change [y,n,a,q,d%s,?]? " ),
16- N_ ("Stage deletion [y,n,a,q,d%s,?]? " ),
17- N_ ("Stage this hunk [y,n,a,q,d%s,?]? " )
14+ struct patch_mode {
15+ const char * diff [4 ], * apply [4 ], * apply_check [4 ];
16+ unsigned is_reverse :1 , apply_for_checkout :1 ;
17+ const char * prompt_mode [PROMPT_HUNK + 1 ];
18+ const char * edit_hunk_hint , * help_patch_text ;
19+ };
20+
21+ static struct patch_mode patch_mode_stage = {
22+ .diff = { "diff-files" , NULL },
23+ .apply = { "--cached" , NULL },
24+ .apply_check = { "--cached" , NULL },
25+ .is_reverse = 0 ,
26+ .prompt_mode = {
27+ N_ ("Stage mode change [y,n,q,a,d%s,?]? " ),
28+ N_ ("Stage deletion [y,n,q,a,d%s,?]? " ),
29+ N_ ("Stage this hunk [y,n,q,a,d%s,?]? " )
30+ },
31+ .edit_hunk_hint = N_ ("If the patch applies cleanly, the edited hunk "
32+ "will immediately be marked for staging." ),
33+ .help_patch_text =
34+ N_ ("y - stage this hunk\n"
35+ "n - do not stage this hunk\n"
36+ "q - quit; do not stage this hunk or any of the remaining "
37+ "ones\n"
38+ "a - stage this hunk and all later hunks in the file\n"
39+ "d - do not stage this hunk or any of the later hunks in "
40+ "the file\n" )
1841};
1942
2043struct hunk_header {
@@ -47,6 +70,10 @@ struct add_p_state {
4770 unsigned deleted :1 , mode_change :1 ,binary :1 ;
4871 } * file_diff ;
4972 size_t file_diff_nr ;
73+
74+ /* patch mode */
75+ struct patch_mode * mode ;
76+ const char * revision ;
5077};
5178
5279static void err (struct add_p_state * s , const char * fmt , ...)
@@ -158,9 +185,18 @@ static int parse_diff(struct add_p_state *s, const struct pathspec *ps)
158185 struct hunk * hunk = NULL ;
159186 int res ;
160187
188+ argv_array_pushv (& args , s -> mode -> diff );
189+ if (s -> revision ) {
190+ struct object_id oid ;
191+ argv_array_push (& args ,
192+ /* could be on an unborn branch */
193+ !strcmp ("HEAD" , s -> revision ) &&
194+ get_oid ("HEAD" , & oid ) ?
195+ empty_tree_oid_hex () : s -> revision );
196+ }
197+ color_arg_index = args .argc ;
161198 /* Use `--no-color` explicitly, just in case `diff.color = always`. */
162- argv_array_pushl (& args , "diff-files" , "-p" , "--no-color" , "--" , NULL );
163- color_arg_index = args .argc - 2 ;
199+ argv_array_pushl (& args , "--no-color" , "-p" , "--" , NULL );
164200 for (i = 0 ; i < ps -> nr ; i ++ )
165201 argv_array_push (& args , ps -> items [i ].original );
166202
@@ -737,11 +773,10 @@ static int edit_hunk_manually(struct add_p_state *s, struct hunk *hunk)
737773 "(context).\n"
738774 "To remove '%c' lines, delete them.\n"
739775 "Lines starting with %c will be removed.\n" ),
740- '-' , '+' , comment_line_char );
741- strbuf_commented_addf (& buf ,
742- _ ("If the patch applies cleanly, the edited hunk "
743- "will immediately be\n"
744- "marked for staging.\n" ));
776+ s -> mode -> is_reverse ? '+' : '-' ,
777+ s -> mode -> is_reverse ? '-' : '+' ,
778+ comment_line_char );
779+ strbuf_commented_addf (& buf , "%s" , _ (s -> mode -> edit_hunk_hint ));
745780 /*
746781 * TRANSLATORS: 'it' refers to the patch mentioned in the previous
747782 * messages.
@@ -837,7 +872,8 @@ static int run_apply_check(struct add_p_state *s,
837872 reassemble_patch (s , file_diff , 1 , & s -> buf );
838873
839874 setup_child_process (& cp , s ,
840- "apply" , "--cached" , "--check" , NULL );
875+ "apply" , "--check" , NULL );
876+ argv_array_pushv (& cp .args , s -> mode -> apply_check );
841877 if (pipe_command (& cp , s -> buf .buf , s -> buf .len , NULL , 0 , NULL , 0 ))
842878 return error (_ ("'git apply --cached' failed" ));
843879
@@ -952,13 +988,6 @@ static size_t display_hunks(struct add_p_state *s,
952988 return end_index ;
953989}
954990
955- static const char help_patch_text [] =
956- N_ ("y - stage this hunk\n"
957- "n - do not stage this hunk\n"
958- "q - quit; do not stage this hunk or any of the remaining ones\n"
959- "a - stage this and all the remaining hunks\n"
960- "d - do not stage this hunk nor any of the remaining hunks\n" );
961-
962991static const char help_patch_remainder [] =
963992N_ ("j - leave this hunk undecided, see next undecided hunk\n"
964993 "J - leave this hunk undecided, see next hunk\n"
@@ -1040,7 +1069,8 @@ static int patch_update_file(struct add_p_state *s,
10401069 prompt_mode_type = PROMPT_HUNK ;
10411070
10421071 color_fprintf (stdout , s -> s .prompt_color ,
1043- _ (prompt_mode [prompt_mode_type ]), s -> buf .buf );
1072+ _ (s -> mode -> prompt_mode [prompt_mode_type ]),
1073+ s -> buf .buf );
10441074 fflush (stdout );
10451075 if (strbuf_getline (& s -> answer , stdin ) == EOF )
10461076 break ;
@@ -1197,7 +1227,7 @@ static int patch_update_file(struct add_p_state *s,
11971227 const char * p = _ (help_patch_remainder ), * eol = p ;
11981228
11991229 color_fprintf (stdout , s -> s .help_color , "%s" ,
1200- _ (help_patch_text ));
1230+ _ (s -> mode -> help_patch_text ));
12011231
12021232 /*
12031233 * Show only those lines of the remainder that are
@@ -1230,18 +1260,20 @@ static int patch_update_file(struct add_p_state *s,
12301260 strbuf_reset (& s -> buf );
12311261 reassemble_patch (s , file_diff , 0 , & s -> buf );
12321262
1233- setup_child_process (& cp , s , "apply" , "--cached" , NULL );
1263+ setup_child_process (& cp , s , "apply" , NULL );
1264+ argv_array_pushv (& cp .args , s -> mode -> apply );
12341265 if (pipe_command (& cp , s -> buf .buf , s -> buf .len ,
12351266 NULL , 0 , NULL , 0 ))
1236- error (_ ("'git apply --cached ' failed" ));
1267+ error (_ ("'git apply' failed" ));
12371268 repo_refresh_and_write_index (s -> s .r , REFRESH_QUIET , 0 );
12381269 }
12391270
12401271 putchar ('\n' );
12411272 return quit ;
12421273}
12431274
1244- int run_add_p (struct repository * r , const struct pathspec * ps )
1275+ int run_add_p (struct repository * r , enum add_p_mode mode ,
1276+ const char * revision , const struct pathspec * ps )
12451277{
12461278 struct add_p_state s = {
12471279 { r }, STRBUF_INIT , STRBUF_INIT , STRBUF_INIT , STRBUF_INIT
@@ -1251,6 +1283,9 @@ int run_add_p(struct repository *r, const struct pathspec *ps)
12511283 if (init_add_i_state (r , & s .s ))
12521284 return error ("Could not read `add -i` config" );
12531285
1286+ s .mode = & patch_mode_stage ;
1287+ s .revision = revision ;
1288+
12541289 if (repo_refresh_and_write_index (r , REFRESH_QUIET , 0 ) < 0 ||
12551290 parse_diff (& s , ps ) < 0 ) {
12561291 strbuf_release (& s .plain );
0 commit comments