Code

Fix waiting for input after executing a run request in pager mode
[tig.git] / tig.c
diff --git a/tig.c b/tig.c
index 26cd2ef386a15f58754a6f8ff915e51578047e40..3394d407472d3317d9cf9cc9d311db445706944b 100644 (file)
--- a/tig.c
+++ b/tig.c
@@ -68,6 +68,7 @@ static int read_properties(FILE *pipe, const char *separators, int (*read)(char
 static void set_nonblocking_input(bool loading);
 static size_t utf8_length(const char *string, int *width, size_t max_width, int *trimmed, bool reserve);
 static bool prompt_yesno(const char *prompt);
+static int load_refs(void);
 
 #define ABS(x)         ((x) >= 0  ? (x) : -(x))
 #define MIN(x, y)      ((x) < (y) ? (x) :  (y))
@@ -77,7 +78,8 @@ static bool prompt_yesno(const char *prompt);
 
 #define SIZEOF_STR     1024    /* Default string size. */
 #define SIZEOF_REF     256     /* Size of symbolic or SHA1 ID. */
-#define SIZEOF_REV     41      /* Holds a SHA-1 and an ending NUL */
+#define SIZEOF_REV     41      /* Holds a SHA-1 and an ending NUL. */
+#define SIZEOF_ARG     32      /* Default argument array size. */
 
 /* Revision graph */
 
@@ -114,7 +116,7 @@ static bool prompt_yesno(const char *prompt);
 #define NULL_ID                "0000000000000000000000000000000000000000"
 
 #ifndef GIT_CONFIG
-#define GIT_CONFIG "git config"
+#define GIT_CONFIG "config"
 #endif
 
 #define TIG_LS_REMOTE \
@@ -162,7 +164,7 @@ struct ref {
        unsigned int next:1;    /* For ref lists: are there more refs? */
 };
 
-static struct ref **get_refs(char *id);
+static struct ref **get_refs(const char *id);
 
 struct int_map {
        const char *name;
@@ -275,6 +277,9 @@ string_enum_compare(const char *str1, const char *str2, int len)
        return 0;
 }
 
+#define prefixcmp(str1, str2) \
+       strncmp(str1, str2, STRING_SIZE(str2))
+
 /* Shell quoting
  *
  * NOTE: The following is a slightly modified copy of the git project's shell
@@ -347,6 +352,13 @@ sq_quote(char buf[SIZEOF_STR], size_t bufsize, const char *src)
        REQ_(VIEW_CLOSE,        "Close the current view"), \
        REQ_(QUIT,              "Close all views and quit"), \
        \
+       REQ_GROUP("View specific requests") \
+       REQ_(STATUS_UPDATE,     "Update file status"), \
+       REQ_(STATUS_REVERT,     "Revert file changes"), \
+       REQ_(STATUS_MERGE,      "Merge file using external tool"), \
+       REQ_(STAGE_NEXT,        "Find next chunk to stage"), \
+       REQ_(TREE_PARENT,       "Switch to parent directory in tree view"), \
+       \
        REQ_GROUP("Cursor navigation") \
        REQ_(MOVE_UP,           "Move cursor one line up"), \
        REQ_(MOVE_DOWN,         "Move cursor one line down"), \
@@ -367,22 +379,19 @@ sq_quote(char buf[SIZEOF_STR], size_t bufsize, const char *src)
        REQ_(FIND_NEXT,         "Find next search match"), \
        REQ_(FIND_PREV,         "Find previous search match"), \
        \
+       REQ_GROUP("Option manipulation") \
+       REQ_(TOGGLE_LINENO,     "Toggle line numbers"), \
+       REQ_(TOGGLE_DATE,       "Toggle date display"), \
+       REQ_(TOGGLE_AUTHOR,     "Toggle author display"), \
+       REQ_(TOGGLE_REV_GRAPH,  "Toggle revision graph visualization"), \
+       REQ_(TOGGLE_REFS,       "Toggle reference display (tags/branches)"), \
+       \
        REQ_GROUP("Misc") \
        REQ_(PROMPT,            "Bring up the prompt"), \
        REQ_(SCREEN_REDRAW,     "Redraw the screen"), \
        REQ_(SCREEN_RESIZE,     "Resize the screen"), \
        REQ_(SHOW_VERSION,      "Show version information"), \
        REQ_(STOP_LOADING,      "Stop all loading views"), \
-       REQ_(TOGGLE_LINENO,     "Toggle line numbers"), \
-       REQ_(TOGGLE_DATE,       "Toggle date display"), \
-       REQ_(TOGGLE_AUTHOR,     "Toggle author display"), \
-       REQ_(TOGGLE_REV_GRAPH,  "Toggle revision graph visualization"), \
-       REQ_(TOGGLE_REFS,       "Toggle reference display (tags/branches)"), \
-       REQ_(STATUS_UPDATE,     "Update file status"), \
-       REQ_(STATUS_CHECKOUT,   "Checkout file"), \
-       REQ_(STATUS_MERGE,      "Merge file using external tool"), \
-       REQ_(STAGE_NEXT,        "Find next chunk to stage"), \
-       REQ_(TREE_PARENT,       "Switch to parent directory in tree view"), \
        REQ_(EDIT,              "Open in editor"), \
        REQ_(NONE,              "Do nothing")
 
@@ -402,9 +411,9 @@ enum request {
 
 struct request_info {
        enum request request;
-       char *name;
+       const char *name;
        int namelen;
-       char *help;
+       const char *help;
 };
 
 static struct request_info req_info[] = {
@@ -474,13 +483,14 @@ static char opt_cdup[SIZEOF_STR]  = "";
 static char opt_git_dir[SIZEOF_STR]    = "";
 static signed char opt_is_inside_work_tree     = -1; /* set to TRUE or FALSE */
 static char opt_editor[SIZEOF_STR]     = "";
+static FILE *opt_tty                   = NULL;
 
 static enum request
-parse_options(int argc, char *argv[])
+parse_options(int argc, const char *argv[])
 {
        enum request request = REQ_VIEW_MAIN;
        size_t buf_size;
-       char *subcommand;
+       const char *subcommand;
        bool seen_dashdash = FALSE;
        int i;
 
@@ -534,7 +544,7 @@ parse_options(int argc, char *argv[])
        buf_size = strlen(opt_cmd);
 
        for (i = 1 + !!subcommand; i < argc; i++) {
-               char *opt = argv[i];
+               const char *opt = argv[i];
 
                if (seen_dashdash || !strcmp(opt, "--")) {
                        seen_dashdash = TRUE;
@@ -644,7 +654,7 @@ static struct line_info line_info[] = {
 };
 
 static enum line_type
-get_line_type(char *line)
+get_line_type(const char *line)
 {
        int linelen = strlen(line);
        enum line_type type;
@@ -666,7 +676,7 @@ get_line_attr(enum line_type type)
 }
 
 static struct line_info *
-get_line_info(char *name)
+get_line_info(const char *name)
 {
        size_t namelen = strlen(name);
        enum line_type type;
@@ -781,7 +791,7 @@ static struct keybinding default_keybindings[] = {
        { 'F',          REQ_TOGGLE_REFS },
        { ':',          REQ_PROMPT },
        { 'u',          REQ_STATUS_UPDATE },
-       { '!',          REQ_STATUS_CHECKOUT },
+       { '!',          REQ_STATUS_REVERT },
        { 'M',          REQ_STATUS_MERGE },
        { '@',          REQ_STAGE_NEXT },
        { ',',          REQ_TREE_PARENT },
@@ -861,7 +871,7 @@ get_keybinding(enum keymap keymap, int key)
 
 
 struct key {
-       char *name;
+       const char *name;
        int value;
 };
 
@@ -911,11 +921,11 @@ get_key_value(const char *name)
        return ERR;
 }
 
-static char *
+static const char *
 get_key_name(int key_value)
 {
        static char key_char[] = "'X'";
-       char *seq = NULL;
+       const char *seq = NULL;
        int key;
 
        for (key = 0; key < ARRAY_SIZE(key_table); key++)
@@ -929,10 +939,10 @@ get_key_name(int key_value)
                seq = key_char;
        }
 
-       return seq ? seq : "'?'";
+       return seq ? seq : "(no key)";
 }
 
-static char *
+static const char *
 get_key(enum request request)
 {
        static char buf[BUFSIZ];
@@ -967,7 +977,7 @@ static struct run_request *run_request;
 static size_t run_requests;
 
 static enum request
-add_run_request(enum keymap keymap, int key, int argc, char **argv)
+add_run_request(enum keymap keymap, int key, int argc, const char **argv)
 {
        struct run_request *req;
        char cmd[SIZEOF_STR];
@@ -1004,7 +1014,7 @@ add_builtin_run_requests(void)
        struct {
                enum keymap keymap;
                int key;
-               char *argv[1];
+               const char *argv[1];
        } reqs[] = {
                { KEYMAP_MAIN,    'C', { "git cherry-pick %(commit)" } },
                { KEYMAP_GENERIC, 'G', { "git gc" } },
@@ -1056,11 +1066,11 @@ static struct int_map attr_map[] = {
 
 static int   config_lineno;
 static bool  config_errors;
-static char *config_msg;
+static const char *config_msg;
 
 /* Wants: object fgcolor bgcolor [attr] */
 static int
-option_color_command(int argc, char *argv[])
+option_color_command(int argc, const char *argv[])
 {
        struct line_info *info;
 
@@ -1113,7 +1123,7 @@ parse_int(const char *s, int default_value, int min, int max)
 
 /* Wants: name = value */
 static int
-option_set_command(int argc, char *argv[])
+option_set_command(int argc, const char *argv[])
 {
        if (argc != 3) {
                config_msg = "Wrong number of arguments given to set command";
@@ -1171,18 +1181,17 @@ option_set_command(int argc, char *argv[])
        }
 
        if (!strcmp(argv[0], "commit-encoding")) {
-               char *arg = argv[2];
-               int delimiter = *arg;
-               int i;
+               const char *arg = argv[2];
+               int arglen = strlen(arg);
 
-               switch (delimiter) {
+               switch (arg[0]) {
                case '"':
                case '\'':
-                       for (arg++, i = 0; arg[i]; i++)
-                               if (arg[i] == delimiter) {
-                                       arg[i] = 0;
-                                       break;
-                               }
+                       if (arglen == 1 || arg[arglen - 1] != arg[0]) {
+                               config_msg = "Unmatched quotation";
+                               return ERR;
+                       }
+                       arg += 1; arglen -= 2;
                default:
                        string_ncopy(opt_encoding, arg, strlen(arg));
                        return OK;
@@ -1195,7 +1204,7 @@ option_set_command(int argc, char *argv[])
 
 /* Wants: mode request key */
 static int
-option_bind_command(int argc, char *argv[])
+option_bind_command(int argc, const char *argv[])
 {
        enum request request;
        int keymap;
@@ -1244,9 +1253,9 @@ option_bind_command(int argc, char *argv[])
 }
 
 static int
-set_option(char *opt, char *value)
+set_option(const char *opt, char *value)
 {
-       char *argv[16];
+       const char *argv[SIZEOF_ARG];
        int valuelen;
        int argc = 0;
 
@@ -1338,9 +1347,9 @@ load_option_file(const char *path)
 static int
 load_options(void)
 {
-       char *home = getenv("HOME");
-       char *tigrc_user = getenv("TIGRC_USER");
-       char *tigrc_system = getenv("TIGRC_SYSTEM");
+       const char *home = getenv("HOME");
+       const char *tigrc_user = getenv("TIGRC_USER");
+       const char *tigrc_system = getenv("TIGRC_SYSTEM");
        char buf[SIZEOF_STR];
 
        add_builtin_run_requests();
@@ -1452,14 +1461,15 @@ struct view_ops {
        void (*select)(struct view *view, struct line *line);
 };
 
-static struct view_ops pager_ops;
-static struct view_ops main_ops;
-static struct view_ops tree_ops;
-static struct view_ops blob_ops;
 static struct view_ops blame_ops;
+static struct view_ops blob_ops;
 static struct view_ops help_ops;
-static struct view_ops status_ops;
+static struct view_ops log_ops;
+static struct view_ops main_ops;
+static struct view_ops pager_ops;
 static struct view_ops stage_ops;
+static struct view_ops status_ops;
+static struct view_ops tree_ops;
 
 #define VIEW_STR(name, cmd, env, ref, ops, map, git) \
        { name, cmd, #env, ref, ops, map, git }
@@ -1471,7 +1481,7 @@ static struct view_ops stage_ops;
 static struct view views[] = {
        VIEW_(MAIN,   "main",   &main_ops,   TRUE,  ref_head),
        VIEW_(DIFF,   "diff",   &pager_ops,  TRUE,  ref_commit),
-       VIEW_(LOG,    "log",    &pager_ops,  TRUE,  ref_head),
+       VIEW_(LOG,    "log",    &log_ops,    TRUE,  ref_head),
        VIEW_(TREE,   "tree",   &tree_ops,   TRUE,  ref_commit),
        VIEW_(BLOB,   "blob",   &blob_ops,   TRUE,  ref_blob),
        VIEW_(BLAME,  "blame",  &blame_ops,  TRUE,  ref_commit),
@@ -1633,7 +1643,7 @@ draw_graphic(struct view *view, enum line_type type, chtype graphic[], size_t si
 }
 
 static bool
-draw_field(struct view *view, enum line_type type, char *text, int len, bool trim)
+draw_field(struct view *view, enum line_type type, const char *text, int len, bool trim)
 {
        int max = MIN(view->width - view->col, len);
        int col;
@@ -2178,6 +2188,24 @@ search_view(struct view *view, enum request request)
  * Incremental updating
  */
 
+static void
+reset_view(struct view *view)
+{
+       int i;
+
+       for (i = 0; i < view->lines; i++)
+               free(view->line[i].data);
+       free(view->line);
+
+       view->line = NULL;
+       view->offset = 0;
+       view->lines  = 0;
+       view->lineno = 0;
+       view->line_size = 0;
+       view->line_alloc = 0;
+       view->vid[0] = 0;
+}
+
 static void
 end_update(struct view *view, bool force)
 {
@@ -2195,7 +2223,7 @@ end_update(struct view *view, bool force)
 }
 
 static bool
-begin_update(struct view *view)
+begin_update(struct view *view, bool refresh)
 {
        if (opt_cmd[0]) {
                string_copy(view->cmd, opt_cmd);
@@ -2220,7 +2248,7 @@ begin_update(struct view *view)
                if (!string_format(view->cmd, format, view->id, path))
                        return FALSE;
 
-       } else {
+       } else if (!refresh) {
                const char *format = view->cmd_env ? view->cmd_env : view->cmd_fmt;
                const char *id = view->id;
 
@@ -2246,23 +2274,9 @@ begin_update(struct view *view)
                return FALSE;
 
        set_nonblocking_input(TRUE);
-
-       view->offset = 0;
-       view->lines  = 0;
-       view->lineno = 0;
+       reset_view(view);
        string_copy_rev(view->vid, view->id);
 
-       if (view->line) {
-               int i;
-
-               for (i = 0; i < view->lines; i++)
-                       if (view->line[i].data)
-                               free(view->line[i].data);
-
-               free(view->line);
-               view->line = NULL;
-       }
-
        view->start_time = time(NULL);
 
        return TRUE;
@@ -2365,8 +2379,14 @@ update_view(struct view *view)
                }
        }
 
-       if (!view_is_displayed(view))
-               goto check_pipe;
+       if (ferror(view->pipe) && errno != 0) {
+               report("Failed to read: %s", strerror(errno));
+               end_update(view, TRUE);
+
+       } else if (feof(view->pipe)) {
+               report("");
+               end_update(view, FALSE);
+       }
 
        if (view == VIEW(REQ_VIEW_TREE)) {
                /* Clear the view and redraw everything since the tree sorting
@@ -2396,17 +2416,6 @@ update_view(struct view *view)
        /* Update the title _after_ the redraw so that if the redraw picks up a
         * commit reference in view->ref it'll be available here. */
        update_view_title(view);
-
-check_pipe:
-       if (ferror(view->pipe) && errno != 0) {
-               report("Failed to read: %s", strerror(errno));
-               end_update(view, TRUE);
-
-       } else if (feof(view->pipe)) {
-               report("");
-               end_update(view, FALSE);
-       }
-
        return TRUE;
 
 alloc_error:
@@ -2428,10 +2437,9 @@ add_line_data(struct view *view, void *data, enum line_type type)
 }
 
 static struct line *
-add_line_text(struct view *view, char *data, enum line_type type)
+add_line_text(struct view *view, const char *text, enum line_type type)
 {
-       if (data)
-               data = strdup(data);
+       char *data = text ? strdup(text) : NULL;
 
        return data ? add_line_data(view, data, type) : NULL;
 }
@@ -2446,7 +2454,8 @@ enum open_flags {
        OPEN_SPLIT = 1,         /* Split current view. */
        OPEN_BACKGROUNDED = 2,  /* Backgrounded. */
        OPEN_RELOAD = 4,        /* Reload view even if it is the current. */
-       OPEN_NOMAXIMIZE = 8     /* Do not maximize the current view. */
+       OPEN_NOMAXIMIZE = 8,    /* Do not maximize the current view. */
+       OPEN_REFRESH = 16,      /* Refresh view using previous command. */
 };
 
 static void
@@ -2454,8 +2463,8 @@ open_view(struct view *prev, enum request request, enum open_flags flags)
 {
        bool backgrounded = !!(flags & OPEN_BACKGROUNDED);
        bool split = !!(flags & OPEN_SPLIT);
-       bool reload = !!(flags & OPEN_RELOAD);
-       bool nomaximize = !!(flags & OPEN_NOMAXIMIZE);
+       bool reload = !!(flags & (OPEN_RELOAD | OPEN_REFRESH));
+       bool nomaximize = !!(flags & (OPEN_NOMAXIMIZE | OPEN_REFRESH));
        struct view *view = VIEW(request);
        int nviews = displayed_views();
        struct view *base_view = display[0];
@@ -2497,7 +2506,7 @@ open_view(struct view *prev, enum request request, enum open_flags flags)
                }
 
        } else if ((reload || strcmp(view->vid, view->id)) &&
-                  !begin_update(view)) {
+                  !begin_update(view, flags & OPEN_REFRESH)) {
                report("Failed to load %s view", view->name);
                return;
        }
@@ -2525,7 +2534,7 @@ open_view(struct view *prev, enum request request, enum open_flags flags)
                 * the screen. */
                werase(view->win);
                report("");
-       } else {
+       } else if (view_is_displayed(view)) {
                redraw_view(view);
                report("");
        }
@@ -2536,12 +2545,15 @@ open_view(struct view *prev, enum request request, enum open_flags flags)
                update_view_title(view);
 }
 
-static void
+static bool
 run_confirm(const char *cmd, const char *prompt)
 {
-       if (prompt_yesno(prompt)) {
+       bool confirmation = prompt_yesno(prompt);
+
+       if (confirmation)
                system(cmd);
-       }
+
+       return confirmation;
 }
 
 static void
@@ -2551,7 +2563,7 @@ open_external_viewer(const char *cmd)
        endwin();                  /* restore original tty modes */
        system(cmd);
        fprintf(stderr, "Press Enter to continue");
-       getc(stdin);
+       getc(opt_tty);
        reset_prog_mode();
        redraw_display();
 }
@@ -2573,7 +2585,7 @@ open_editor(bool from_root, const char *file)
 {
        char cmd[SIZEOF_STR];
        char file_sq[SIZEOF_STR];
-       char *editor;
+       const char *editor;
        char *prefix = from_root ? opt_cdup : "";
 
        editor = getenv("GIT_EDITOR");
@@ -2661,6 +2673,7 @@ view_driver(struct view *view, enum request request)
                /* FIXME: When all views can refresh always do this. */
                if (view == VIEW(REQ_VIEW_STATUS) ||
                    view == VIEW(REQ_VIEW_MAIN) ||
+                   view == VIEW(REQ_VIEW_LOG) ||
                    view == VIEW(REQ_VIEW_STAGE))
                        request = REQ_REFRESH;
                else
@@ -2870,6 +2883,7 @@ view_driver(struct view *view, enum request request)
                        view->parent = view;
                        resize_display();
                        redraw_display();
+                       report("");
                        break;
                }
                /* Fall-through */
@@ -2877,7 +2891,6 @@ view_driver(struct view *view, enum request request)
                return FALSE;
 
        default:
-               /* An unknown key will show most commonly used commands. */
                report("Unknown key, press 'h' for help");
                return TRUE;
        }
@@ -2903,7 +2916,7 @@ pager_draw(struct view *view, struct line *line, unsigned int lineno)
 }
 
 static bool
-add_describe_ref(char *buf, size_t *bufpos, char *commit_id, const char *sep)
+add_describe_ref(char *buf, size_t *bufpos, const char *commit_id, const char *sep)
 {
        char refbuf[SIZEOF_STR];
        char *ref = NULL;
@@ -2951,8 +2964,8 @@ add_pager_refs(struct view *view, struct line *line)
 
        do {
                struct ref *ref = refs[refpos];
-               char *fmt = ref->tag    ? "%s[%s]" :
-                           ref->remote ? "%s<%s>" : "%s%s";
+               const char *fmt = ref->tag    ? "%s[%s]" :
+                                 ref->remote ? "%s<%s>" : "%s%s";
 
                if (!string_format_from(buf, &bufpos, fmt, sep, ref->name))
                        return;
@@ -3063,6 +3076,29 @@ static struct view_ops pager_ops = {
        pager_select,
 };
 
+static enum request
+log_request(struct view *view, enum request request, struct line *line)
+{
+       switch (request) {
+       case REQ_REFRESH:
+               load_refs();
+               open_view(view, REQ_VIEW_LOG, OPEN_REFRESH);
+               return REQ_NONE;
+       default:
+               return pager_request(view, request, line);
+       }
+}
+
+static struct view_ops log_ops = {
+       "line",
+       NULL,
+       pager_read,
+       pager_draw,
+       log_request,
+       pager_grep,
+       pager_select,
+};
+
 
 /*
  * Help backend
@@ -3091,7 +3127,7 @@ help_open(struct view *view)
        add_line_text(view, "Quick reference for tig keybindings:", LINE_DEFAULT);
 
        for (i = 0; i < ARRAY_SIZE(req_info); i++) {
-               char *key;
+               const char *key;
 
                if (req_info[i].request == REQ_NONE)
                        continue;
@@ -3119,7 +3155,7 @@ help_open(struct view *view)
 
        for (i = 0; i < run_requests; i++) {
                struct run_request *req = get_run_request(REQ_NONE + i + 1);
-               char *key;
+               const char *key;
 
                if (!req)
                        continue;
@@ -3176,7 +3212,7 @@ pop_tree_stack_entry(void)
 }
 
 static void
-push_tree_stack_entry(char *name, unsigned long lineno)
+push_tree_stack_entry(const char *name, unsigned long lineno)
 {
        struct tree_stack_entry *entry = calloc(1, sizeof(*entry));
        size_t pathlen = strlen(opt_path);
@@ -3212,8 +3248,8 @@ push_tree_stack_entry(char *name, unsigned long lineno)
 #define TREE_UP_FORMAT "040000 tree %s\t.."
 
 static int
-tree_compare_entry(enum line_type type1, char *name1,
-                  enum line_type type2, char *name2)
+tree_compare_entry(enum line_type type1, const char *name1,
+                  enum line_type type2, const char *name2)
 {
        if (type1 != type2) {
                if (type1 == LINE_TREE_DIR)
@@ -3224,10 +3260,10 @@ tree_compare_entry(enum line_type type1, char *name1,
        return strcmp(name1, name2);
 }
 
-static char *
+static const char *
 tree_path(struct line *line)
 {
-       char *path = line->data;
+       const char *path = line->data;
 
        return path + SIZEOF_TREE_ATTR;
 }
@@ -3279,7 +3315,7 @@ tree_read(struct view *view, char *text)
        /* Skip "Directory ..." and ".." line. */
        for (pos = 1 + !!*opt_path; pos < view->lines; pos++) {
                struct line *line = &view->line[pos];
-               char *path1 = tree_path(line);
+               const char *path1 = tree_path(line);
                char *path2 = text + SIZEOF_TREE_ATTR;
                int cmp = tree_compare_entry(line->type, path1, type, path2);
 
@@ -3318,7 +3354,7 @@ tree_request(struct view *view, enum request request, struct line *line)
        enum open_flags flags;
 
        if (request == REQ_VIEW_BLAME) {
-               char *filename = tree_path(line);
+               const char *filename = tree_path(line);
 
                if (line->type == LINE_TREE_DIR) {
                        report("Cannot show blame for directory %s", opt_path);
@@ -3354,7 +3390,7 @@ tree_request(struct view *view, enum request request, struct line *line)
                        pop_tree_stack_entry();
 
                } else {
-                       char *basename = tree_path(line);
+                       const char *basename = tree_path(line);
 
                        push_tree_stack_entry(basename, view->lineno);
                }
@@ -3451,7 +3487,7 @@ struct blame {
 };
 
 #define BLAME_CAT_FILE_CMD "git cat-file blob %s:%s"
-#define BLAME_INCREMENTAL_CMD "git blame --incremental %s %s"
+#define BLAME_INCREMENTAL_CMD "git blame --incremental %s -- %s"
 
 static bool
 blame_open(struct view *view)
@@ -3483,21 +3519,10 @@ blame_open(struct view *view)
        if (!string_format(view->cmd, BLAME_INCREMENTAL_CMD, ref, path))
                return FALSE;
 
+       reset_view(view);
        string_format(view->ref, "%s ...", opt_file);
        string_copy_rev(view->vid, opt_file);
        set_nonblocking_input(TRUE);
-
-       if (view->line) {
-               int i;
-
-               for (i = 0; i < view->lines; i++)
-                       free(view->line[i].data);
-               free(view->line);
-       }
-
-       view->lines = view->line_alloc = view->line_size = view->lineno = 0;
-       view->offset = view->lines  = view->lineno = 0;
-       view->line = NULL;
        view->start_time = time(NULL);
 
        return TRUE;
@@ -3528,9 +3553,9 @@ get_blame_commit(struct view *view, const char *id)
 }
 
 static bool
-parse_number(char **posref, size_t *number, size_t min, size_t max)
+parse_number(const char **posref, size_t *number, size_t min, size_t max)
 {
-       char *pos = *posref;
+       const char *pos = *posref;
 
        *posref = NULL;
        pos = strchr(pos + 1, ' ');
@@ -3545,11 +3570,11 @@ parse_number(char **posref, size_t *number, size_t min, size_t max)
 }
 
 static struct blame_commit *
-parse_blame_commit(struct view *view, char *text, int *blamed)
+parse_blame_commit(struct view *view, const char *text, int *blamed)
 {
        struct blame_commit *commit;
        struct blame *blame;
-       char *pos = text + SIZEOF_REV - 1;
+       const char *pos = text + SIZEOF_REV - 1;
        size_t lineno;
        size_t group;
 
@@ -3578,7 +3603,7 @@ parse_blame_commit(struct view *view, char *text, int *blamed)
 }
 
 static bool
-blame_read_file(struct view *view, char *line)
+blame_read_file(struct view *view, const char *line)
 {
        if (!line) {
                FILE *pipe = NULL;
@@ -3683,7 +3708,7 @@ blame_draw(struct view *view, struct line *line, unsigned int lineno)
 {
        struct blame *blame = line->data;
        struct tm *time = NULL;
-       char *id = NULL, *author = NULL;
+       const char *id = NULL, *author = NULL;
 
        if (blame->commit && *blame->commit->filename) {
                id = blame->commit->id;
@@ -3816,17 +3841,24 @@ static enum line_type stage_line_type;
 static size_t stage_chunks;
 static int *stage_chunk;
 
+/* This should work even for the "On branch" line. */
+static inline bool
+status_has_none(struct view *view, struct line *line)
+{
+       return line < view->line + view->lines && !line[1].data;
+}
+
 /* Get fields from the diff line:
  * :100644 100644 06a5d6ae9eca55be2e0e585a152e6b1336f2b20e 0000000000000000000000000000000000000000 M
  */
 static inline bool
-status_get_diff(struct status *file, char *buf, size_t bufsize)
+status_get_diff(struct status *file, const char *buf, size_t bufsize)
 {
-       char *old_mode = buf +  1;
-       char *new_mode = buf +  8;
-       char *old_rev  = buf + 15;
-       char *new_rev  = buf + 56;
-       char *status   = buf + 97;
+       const char *old_mode = buf +  1;
+       const char *new_mode = buf +  8;
+       const char *old_rev  = buf + 15;
+       const char *new_rev  = buf + 56;
+       const char *status   = buf + 97;
 
        if (bufsize < 99 ||
            old_mode[-1] != ':' ||
@@ -3967,9 +3999,9 @@ error_out:
 #define STATUS_DIFF_INDEX_CMD "git diff-index -z --diff-filter=ACDMRTXB --cached -M HEAD"
 #define STATUS_DIFF_FILES_CMD "git diff-files -z"
 #define STATUS_LIST_OTHER_CMD \
-       "git ls-files -z --others --exclude-per-directory=.gitignore"
+       "git ls-files -z --others --exclude-standard"
 #define STATUS_LIST_NO_HEAD_CMD \
-       "git ls-files -z --cached --exclude-per-directory=.gitignore"
+       "git ls-files -z --cached --exclude-standard"
 
 #define STATUS_DIFF_INDEX_SHOW_CMD \
        "git diff-index --root --patch-with-stat -C -M --cached HEAD -- %s %s 2>/dev/null"
@@ -3986,19 +4018,9 @@ error_out:
 static bool
 status_open(struct view *view)
 {
-       struct stat statbuf;
-       char exclude[SIZEOF_STR];
-       char indexcmd[SIZEOF_STR] = STATUS_DIFF_INDEX_CMD;
-       char othercmd[SIZEOF_STR] = STATUS_LIST_OTHER_CMD;
        unsigned long prev_lineno = view->lineno;
-       char indexstatus = 0;
-       size_t i;
 
-       for (i = 0; i < view->lines; i++)
-               free(view->line[i].data);
-       free(view->line);
-       view->lines = view->line_alloc = view->line_size = view->lineno = 0;
-       view->line = NULL;
+       reset_view(view);
 
        if (!realloc_lines(view, view->line_size + 7))
                return FALSE;
@@ -4011,33 +4033,17 @@ status_open(struct view *view)
        else if (!string_format(status_onbranch, "On branch %s", opt_head))
                return FALSE;
 
-       if (opt_no_head) {
-               string_copy(indexcmd, STATUS_LIST_NO_HEAD_CMD);
-               indexstatus = 'A';
-       }
-
-       if (!string_format(exclude, "%s/info/exclude", opt_git_dir))
-               return FALSE;
-
-       if (stat(exclude, &statbuf) >= 0) {
-               size_t cmdsize = strlen(othercmd);
-
-               if (!string_format_from(othercmd, &cmdsize, " %s", "--exclude-from=") ||
-                   sq_quote(othercmd, cmdsize, exclude) >= sizeof(othercmd))
-                       return FALSE;
+       system("git update-index -q --refresh >/dev/null 2>/dev/null");
 
-               cmdsize = strlen(indexcmd);
-               if (opt_no_head &&
-                   (!string_format_from(indexcmd, &cmdsize, " %s", "--exclude-from=") ||
-                    sq_quote(indexcmd, cmdsize, exclude) >= sizeof(indexcmd)))
+       if (opt_no_head) {
+               if (!status_run(view, STATUS_LIST_NO_HEAD_CMD, 'A', LINE_STAT_STAGED))
                        return FALSE;
+       } else if (!status_run(view, STATUS_DIFF_INDEX_CMD, 0, LINE_STAT_STAGED)) {
+               return FALSE;
        }
 
-       system("git update-index -q --refresh >/dev/null 2>/dev/null");
-
-       if (!status_run(view, indexcmd, indexstatus, LINE_STAT_STAGED) ||
-           !status_run(view, STATUS_DIFF_FILES_CMD, 0, LINE_STAT_UNSTAGED) ||
-           !status_run(view, othercmd, '?', LINE_STAT_UNTRACKED))
+       if (!status_run(view, STATUS_DIFF_FILES_CMD, 0, LINE_STAT_UNSTAGED) ||
+           !status_run(view, STATUS_LIST_OTHER_CMD, '?', LINE_STAT_UNTRACKED))
                return FALSE;
 
        /* If all went well restore the previous line number to stay in
@@ -4069,7 +4075,7 @@ status_draw(struct view *view, struct line *line, unsigned int lineno)
 {
        struct status *status = line->data;
        enum line_type type;
-       char *text;
+       const char *text;
 
        if (!status) {
                switch (line->type) {
@@ -4121,7 +4127,7 @@ status_enter(struct view *view, struct line *line)
        struct status *status = line->data;
        char oldpath[SIZEOF_STR] = "";
        char newpath[SIZEOF_STR] = "";
-       char *info;
+       const char *info;
        size_t cmdsize = 0;
        enum open_flags split;
 
@@ -4365,36 +4371,29 @@ status_update(struct view *view)
 }
 
 static bool
-status_checkout(struct view *view)
-{
-       struct line *line = &view->line[view->lineno];
-
-       assert(view->lines);
-
-       if (!line->data || line->type != LINE_STAT_UNSTAGED) {
-               /* This should work even for the "On branch" line. */
-               if (line < view->line + view->lines && !line[1].data) {
-                       report("Nothing to checkout");
-               } else if (line->type == LINE_STAT_UNTRACKED) {
-                       report("Cannot checkout untracked files");
-               } else if (line->type == LINE_STAT_STAGED) {
-                       report("Cannot checkout staged files");
+status_revert(struct status *status, enum line_type type, bool has_none)
+{
+       if (!status || type != LINE_STAT_UNSTAGED) {
+               if (type == LINE_STAT_STAGED) {
+                       report("Cannot revert changes to staged files");
+               } else if (type == LINE_STAT_UNTRACKED) {
+                       report("Cannot revert changes to untracked files");
+               } else if (has_none) {
+                       report("Nothing to revert");
                } else {
-                       report("Cannot checkout multiple files");
+                       report("Cannot revert changes to multiple files");
                }
                return FALSE;
 
        } else {
-               struct status *status = line->data;
                char cmd[SIZEOF_STR];
                char file_sq[SIZEOF_STR];
 
-               if (sq_quote(file_sq, 0, status->old.name) < sizeof(file_sq) &&
-                   string_format(cmd, "git checkout %s%s", opt_cdup, file_sq)) {
-                       run_confirm(cmd, "Are you sure you want to overwrite any changes?");
-               }
+               if (sq_quote(file_sq, 0, status->old.name) >= sizeof(file_sq) ||
+                   !string_format(cmd, "git checkout -- %s%s", opt_cdup, file_sq))
+                       return FALSE;
 
-               return TRUE;
+               return run_confirm(cmd, "Are you sure you want to overwrite any changes?");
        }
 }
 
@@ -4409,8 +4408,8 @@ status_request(struct view *view, enum request request, struct line *line)
                        return REQ_NONE;
                break;
 
-       case REQ_STATUS_CHECKOUT:
-               if (!status_checkout(view))
+       case REQ_STATUS_REVERT:
+               if (!status_revert(status, line->type, status_has_none(view, line)))
                        return REQ_NONE;
                break;
 
@@ -4461,8 +4460,8 @@ status_select(struct view *view, struct line *line)
 {
        struct status *status = line->data;
        char file[SIZEOF_STR] = "all files";
-       char *text;
-       char *key;
+       const char *text;
+       const char *key;
 
        if (status && !string_format(file, "'%s'", status->new.name))
                return;
@@ -4515,7 +4514,7 @@ status_grep(struct view *view, struct line *line)
                return FALSE;
 
        for (state = S_STATUS; state < S_END; state++) {
-               char *text;
+               const char *text;
 
                switch (state) {
                case S_NAME:    text = status->new.name;        break;
@@ -4549,7 +4548,7 @@ static struct view_ops status_ops = {
 static bool
 stage_diff_line(FILE *pipe, struct line *line)
 {
-       char *buf = line->data;
+       const char *buf = line->data;
        size_t bufsize = strlen(buf);
        size_t written = 0;
 
@@ -4587,7 +4586,7 @@ stage_diff_find(struct view *view, struct line *line, enum line_type type)
 }
 
 static bool
-stage_update_chunk(struct view *view, struct line *chunk)
+stage_apply_chunk(struct view *view, struct line *chunk, bool revert)
 {
        char cmd[SIZEOF_STR];
        size_t cmdsize = 0;
@@ -4603,9 +4602,10 @@ stage_update_chunk(struct view *view, struct line *chunk)
                return FALSE;
 
        if (!string_format_from(cmd, &cmdsize,
-                               "git apply --whitespace=nowarn --cached %s - && "
+                               "git apply --whitespace=nowarn %s %s - && "
                                "git update-index -q --unmerged --refresh 2>/dev/null",
-                               stage_line_type == LINE_STAT_STAGED ? "-R" : ""))
+                               revert ? "" : "--cached",
+                               revert || stage_line_type == LINE_STAT_STAGED ? "-R" : ""))
                return FALSE;
 
        pipe = popen(cmd, "w");
@@ -4630,7 +4630,7 @@ stage_update(struct view *view, struct line *line)
                chunk = stage_diff_find(view, line, LINE_DIFF_CHUNK);
 
        if (chunk) {
-               if (!stage_update_chunk(view, chunk)) {
+               if (!stage_apply_chunk(view, chunk, FALSE)) {
                        report("Failed to apply chunk");
                        return FALSE;
                }
@@ -4655,6 +4655,31 @@ stage_update(struct view *view, struct line *line)
        return TRUE;
 }
 
+static bool
+stage_revert(struct view *view, struct line *line)
+{
+       struct line *chunk = NULL;
+
+       if (!opt_no_head && stage_line_type == LINE_STAT_UNSTAGED)
+               chunk = stage_diff_find(view, line, LINE_DIFF_CHUNK);
+
+       if (chunk) {
+               if (!prompt_yesno("Are you sure you want to revert changes?"))
+                       return FALSE;
+
+               if (!stage_apply_chunk(view, chunk, TRUE)) {
+                       report("Failed to revert chunk");
+                       return FALSE;
+               }
+               return TRUE;
+
+       } else {
+               return status_revert(stage_status.status ? &stage_status : NULL,
+                                    stage_line_type, FALSE);
+       }
+}
+
+
 static void
 stage_next(struct view *view, struct line *line)
 {
@@ -4700,6 +4725,11 @@ stage_request(struct view *view, enum request request, struct line *line)
                        return REQ_NONE;
                break;
 
+       case REQ_STATUS_REVERT:
+               if (!stage_revert(view, line))
+                       return REQ_NONE;
+               break;
+
        case REQ_STAGE_NEXT:
                if (stage_line_type == LINE_STAT_UNTRACKED) {
                        report("File is untracked; press %s to add",
@@ -4743,9 +4773,7 @@ stage_request(struct view *view, enum request request, struct line *line)
 
        if (stage_line_type == LINE_STAT_UNTRACKED)
                opt_pipe = fopen(stage_status.new.name, "r");
-       else
-               string_copy(opt_cmd, view->cmd);
-       open_view(view, REQ_VIEW_STAGE, OPEN_RELOAD | OPEN_NOMAXIMIZE);
+       open_view(view, REQ_VIEW_STAGE, OPEN_REFRESH);
 
        return REQ_NONE;
 }
@@ -4843,7 +4871,7 @@ done_rev_graph(struct rev_graph *graph)
 }
 
 static void
-push_rev_graph(struct rev_graph *graph, char *parent)
+push_rev_graph(struct rev_graph *graph, const char *parent)
 {
        int i;
 
@@ -5182,8 +5210,8 @@ main_request(struct view *view, enum request request, struct line *line)
                open_view(view, REQ_VIEW_DIFF, flags);
                break;
        case REQ_REFRESH:
-               string_copy(opt_cmd, view->cmd);
-               open_view(view, REQ_VIEW_MAIN, OPEN_RELOAD);
+               load_refs();
+               open_view(view, REQ_VIEW_MAIN, OPEN_REFRESH);
                break;
        default:
                return request;
@@ -5500,13 +5528,13 @@ init_display(void)
        /* Initialize the curses library */
        if (isatty(STDIN_FILENO)) {
                cursed = !!initscr();
+               opt_tty = stdin;
        } else {
                /* Leave stdin and stdout alone when acting as a pager. */
-               FILE *io = fopen("/dev/tty", "r+");
-
-               if (!io)
+               opt_tty = fopen("/dev/tty", "r+");
+               if (!opt_tty)
                        die("Failed to open /dev/tty");
-               cursed = !!newterm(NULL, io, io);
+               cursed = !!newterm(NULL, opt_tty, opt_tty);
        }
 
        if (!cursed)
@@ -5667,8 +5695,27 @@ static struct ref ***id_refs = NULL;
 static size_t id_refs_alloc = 0;
 static size_t id_refs_size = 0;
 
+static int
+compare_refs(const void *ref1_, const void *ref2_)
+{
+       const struct ref *ref1 = *(const struct ref **)ref1_;
+       const struct ref *ref2 = *(const struct ref **)ref2_;
+
+       if (ref1->tag != ref2->tag)
+               return ref2->tag - ref1->tag;
+       if (ref1->ltag != ref2->ltag)
+               return ref2->ltag - ref2->ltag;
+       if (ref1->head != ref2->head)
+               return ref2->head - ref1->head;
+       if (ref1->tracked != ref2->tracked)
+               return ref2->tracked - ref1->tracked;
+       if (ref1->remote != ref2->remote)
+               return ref2->remote - ref1->remote;
+       return strcmp(ref1->name, ref2->name);
+}
+
 static struct ref **
-get_refs(char *id)
+get_refs(const char *id)
 {
        struct ref ***tmp_id_refs;
        struct ref **ref_list = NULL;
@@ -5702,19 +5749,20 @@ get_refs(char *id)
                }
 
                ref_list = tmp;
-               if (ref_list_size > 0)
-                       ref_list[ref_list_size - 1]->next = 1;
                ref_list[ref_list_size] = &refs[i];
-
                /* XXX: The properties of the commit chains ensures that we can
                 * safely modify the shared ref. The repo references will
                 * always be similar for the same id. */
-               ref_list[ref_list_size]->next = 0;
+               ref_list[ref_list_size]->next = 1;
+
                ref_list_size++;
        }
 
-       if (ref_list)
+       if (ref_list) {
+               qsort(ref_list, ref_list_size, sizeof(*ref_list), compare_refs);
+               ref_list[ref_list_size - 1]->next = 0;
                id_refs[id_refs_size++] = ref_list;
+       }
 
        return ref_list;
 }
@@ -5730,7 +5778,7 @@ read_ref(char *id, size_t idlen, char *name, size_t namelen)
        bool check_replace = FALSE;
        bool head = FALSE;
 
-       if (!strncmp(name, "refs/tags/", STRING_SIZE("refs/tags/"))) {
+       if (!prefixcmp(name, "refs/tags/")) {
                if (!strcmp(name + namelen - 3, "^{}")) {
                        namelen -= 3;
                        name[namelen] = 0;
@@ -5744,13 +5792,13 @@ read_ref(char *id, size_t idlen, char *name, size_t namelen)
                namelen -= STRING_SIZE("refs/tags/");
                name    += STRING_SIZE("refs/tags/");
 
-       } else if (!strncmp(name, "refs/remotes/", STRING_SIZE("refs/remotes/"))) {
+       } else if (!prefixcmp(name, "refs/remotes/")) {
                remote = TRUE;
                namelen -= STRING_SIZE("refs/remotes/");
                name    += STRING_SIZE("refs/remotes/");
                tracked  = !strcmp(opt_remote, name);
 
-       } else if (!strncmp(name, "refs/heads/", STRING_SIZE("refs/heads/"))) {
+       } else if (!prefixcmp(name, "refs/heads/")) {
                namelen -= STRING_SIZE("refs/heads/");
                name    += STRING_SIZE("refs/heads/");
                head     = !strncmp(opt_head, name, namelen);
@@ -5763,7 +5811,7 @@ read_ref(char *id, size_t idlen, char *name, size_t namelen)
        if (check_replace && !strcmp(name, refs[refs_size - 1].name)) {
                /* it's an annotated tag, replace the previous sha1 with the
                 * resolved commit id; relies on the fact git-ls-remote lists
-                * the commit id of an annotated tag right beofre the commit id
+                * the commit id of an annotated tag right before the commit id
                 * it points to. */
                refs[refs_size - 1].ltag = ltag;
                string_copy_rev(refs[refs_size - 1].id, id);
@@ -5797,6 +5845,14 @@ load_refs(void)
        const char *cmd_env = getenv("TIG_LS_REMOTE");
        const char *cmd = cmd_env && *cmd_env ? cmd_env : TIG_LS_REMOTE;
 
+       if (!*opt_git_dir)
+               return OK;
+
+       while (refs_size > 0)
+               free(refs[--refs_size].name);
+       while (id_refs_size > 0)
+               free(id_refs[--id_refs_size]);
+
        return read_properties(popen(cmd, "r"), "\t", read_ref);
 }
 
@@ -5822,7 +5878,7 @@ read_repo_config_option(char *name, size_t namelen, char *value, size_t valuelen
            !strcmp(name + 7 + strlen(opt_head), ".merge")) {
                size_t from = strlen(opt_remote);
 
-               if (!strncmp(value, "refs/heads/", STRING_SIZE("refs/heads/"))) {
+               if (!prefixcmp(value, "refs/heads/")) {
                        value += STRING_SIZE("refs/heads/");
                        valuelen -= STRING_SIZE("refs/heads/");
                }
@@ -5837,7 +5893,7 @@ read_repo_config_option(char *name, size_t namelen, char *value, size_t valuelen
 static int
 load_git_config(void)
 {
-       return read_properties(popen(GIT_CONFIG " --list", "r"),
+       return read_properties(popen("git " GIT_CONFIG " --list", "r"),
                               "=", read_repo_config_option);
 }
 
@@ -5858,7 +5914,7 @@ read_repo_info(char *name, size_t namelen, char *value, size_t valuelen)
        } else if (opt_cdup[0] == ' ') {
                string_ncopy(opt_cdup, name, namelen);
        } else {
-               if (!strncmp(name, "refs/heads/", STRING_SIZE("refs/heads/"))) {
+               if (!prefixcmp(name, "refs/heads/")) {
                        namelen -= STRING_SIZE("refs/heads/");
                        name    += STRING_SIZE("refs/heads/");
                        string_ncopy(opt_head, name, namelen);
@@ -5970,7 +6026,7 @@ warn(const char *msg, ...)
 }
 
 int
-main(int argc, char *argv[])
+main(int argc, const char *argv[])
 {
        struct view *view;
        enum request request;
@@ -6010,10 +6066,10 @@ main(int argc, char *argv[])
                        die("Failed to initialize character set conversion");
        }
 
-       if (*opt_git_dir && load_refs() == ERR)
+       if (load_refs() == ERR)
                die("Failed to load refs.");
 
-       for (i = 0; i < ARRAY_SIZE(views) && (view = &views[i]); i++)
+       foreach_view (view, i)
                view->cmd_env = getenv(view->cmd_env);
 
        init_display();
@@ -6024,6 +6080,7 @@ main(int argc, char *argv[])
 
                foreach_view (view, i)
                        update_view(view);
+               view = display[current_view];
 
                /* Refresh, accept single keystroke of input */
                key = wgetch(status_win);
@@ -6035,7 +6092,7 @@ main(int argc, char *argv[])
                        continue;
                }
 
-               request = get_keybinding(display[current_view]->keymap, key);
+               request = get_keybinding(view->keymap, key);
 
                /* Some low-level request handling. This keeps access to
                 * status_win restricted. */
@@ -6061,8 +6118,7 @@ main(int argc, char *argv[])
                case REQ_SEARCH:
                case REQ_SEARCH_BACK:
                {
-                       const char *prompt = request == REQ_SEARCH
-                                          ? "/" : "?";
+                       const char *prompt = request == REQ_SEARCH ? "/" : "?";
                        char *search = read_prompt(prompt);
 
                        if (search)