summaryrefslogblamecommitdiffstats
path: root/menu.c
blob: 2bd89adba3cc573738f3de09e02e2849e3ab8d4f (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
  
                                                                    












                                                                           
                                                                                      

   



                    


                        
                  


                    
 

                              




                                                                      

                                                                           
 
             


                                     
                        




















                                                                                 
 
                                         

            
                           
     
                           
                                      


                   
                                  

                                       



                                     



                                                          
      
                  
                                  

                                       



                                     



                                                          
      
                  
                              

                                       



                                 



                                                      
      
                  
                               

                                       



                                  



                                                       
      
                  
                               

                                       



                                  



                                                       
      
                  
                              

                                       



                                 
                                     
                                                      

                               
      
                  
                              

                                       



                                 
                                     
                                                      

                               
      
                  
                               

                        
                                

                        
                              

                                                    
                                

                        
                                

                        
                                 

                        
         
                 
       
                          
     
                                                             
     
                             
                    
     

            


   
                                                                   
 
                   
   
                                             





                                          
                                                               
 
                                  
                                              
                                          
 
                                                                                          
               
                  



                                      


                                                               
               






                                               

                                                   
                 
   

                                       



                                              


                                 

 



                                        
                                                          
                             

                                            
               


                                 


                                         
                                  
                  


      

                                       
                        
        
                  




                                                         

                            
                                                   
                                                
                                                                         
 
                             
       

                                   
                                       


                        
                                                                 

            









                                                                 


        

                   
                                                                           
     
   
               




                                        
                        
                                  
 
                   



                                   





                                                                      
                                                                                    



                              
                       



                                             
                                                                  

                                                                                        
                                                                  


                                         
                                 
                                                                                             



                                     
                                                                
                                              
                                                                

                                                    
                                            
                                                             
                                              
                                                                                   
                                                                

                                
               



                                         
                        
                                                

                                                                                 
                                                           
                                            


                              
                                 
                  
                  
                
                                              
                                                           

      

                                                           
                                
               

 
                                               
 
                   
   
                           
     
                     





                               

                                                                


   

                                         


                                               
                                                                                           
   
                       
     
                    
                                   
     
   
      
   
                                                                            
     
                                        
                                      
                                                              
                                                          
     
        
     

                                                                                                                       
                                                                
                                                                                                   
     
   
 

                                                                  



                                 








                               
                                  
               
                                                                             
     
                                                                    
       
                                             



                                     
                                              


      
                                





                                    

                                                 


                                                                               

                  
                                                                         



                                  
                                                      

      
                                





                                    

                                                 
                
                                                                            



                                
                                                  

 
  





                                  
                                                          
 


                                               

                


                                                                        
     










                                                                    
                     

                             

                                  
                                                                         
     
                               


                                   
                                                       
                                                    


                                                       

      
                                
 





                                                                      


                                    

                                                                          
 







                                                 




















                                                  
                                














                                                    
                                









                                      
                                









                                     
                                

 







                                      
                                











                                                  
                                











                                                  
                                

 
                                            






                                    
                                               

 
                                            






                                 
                                                

 
                                       
 
                                    

 



                                                                
                                            


                                         








                                  


                                                                


                                       
                 

                 
                
                          




                                     
                           
                                  
 




                                    

        
                   

                                 
                              
 
                          
   
 
                                              

 









                                                            




                                                               








                                                                 
                           



                                           

                      







                                                                                   

                                  
   
                                  



                                                             



                            









                                                                      
   

 









                                              







                                       





                                                     





                                              
                                    






                                                




                                                                        


   
 

                          


                                               
                  
                

                         
                                                              
                                     
 

                                                         
   


                                                                          
                                                                       
                  




                                                         
                                                                 
                                        
   
 
                                                           


                               
                                                                                   

                                         




                                


                                               










                                         




                                            
                
                             


              



                                           
                       
                          
                       
                          
                                               
                         
                                                  
                            

                            
   
 







                                                      



                        
 
                











                                                
                                                            



              

                                





                                    






                                                              
 

                               
 

                                   
                  
                                    

                               





                                                                  
 

                              
 


                 


                                  


         




                                   







                                                                        

                      













                                                           


                                      

                                        







                                                                        

                                     
 
                                        
                                
                                                                                     
                                         
                                                                                     
        

                                                                                 

                    
 
                                             

                                                          
 
                              
                                                      
     






                                                    

                       
                            
                 
       
                                  



                                           
                                                
       
                                       



                                         

                                                 


                      
              


                                                    
               
     
 



                                               
                     
                                       






















































                                   
                                                                             







                                                            
                                                                   


                   



                                                                  



                              


                  
                                       
         
                                                      
           

                                                     
                                         



                                                        

                                                                          

                              
                                                    

                
                                             

              
                                        

            
                                                    



                             

              



                         



                            









                                   




                                      



                                  


                       





                   
/*
 * Copyright (C) 1996-2000,2002,2012 Michael R. Elkins <me@mutt.org>
 *
 *     This program is free software; you can redistribute it and/or modify
 *     it under the terms of the GNU General Public License as published by
 *     the Free Software Foundation; either version 2 of the License, or
 *     (at your option) any later version.
 *
 *     This program is distributed in the hope that it will be useful,
 *     but WITHOUT ANY WARRANTY; without even the implied warranty of
 *     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *     GNU General Public License for more details.
 *
 *     You should have received a copy of the GNU General Public License
 *     along with this program; if not, write to the Free Software
 *     Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 */

#if HAVE_CONFIG_H
# include "config.h"
#endif

#include "mutt.h"
#include "mutt_curses.h"
#include "mutt_menu.h"
#include "mbyte.h"
#ifdef USE_SIDEBAR
#include "sidebar.h"
#endif

char* SearchBuffers[MENU_MAX];

/* These are used to track the active menus, for redraw operations. */
static size_t MenuStackCount = 0;
static size_t MenuStackLen = 0;
static MUTTMENU **MenuStack = NULL;

static void print_enriched_string (COLOR_ATTR base_color, unsigned char *s,
                                   int use_indicator)
{
  wchar_t wc;
  size_t k;
  size_t n = mutt_strlen ((char *)s);
  mbstate_t mbstate;
  COLOR_ATTR tree_color;

  if (option (OPTCURSOROVERLAY))
  {
    tree_color = mutt_merge_colors (base_color, ColorDefs[MT_COLOR_TREE]);
    if (use_indicator)
    {
      tree_color = mutt_merge_colors (tree_color, ColorDefs[MT_COLOR_INDICATOR]);
      base_color = mutt_merge_colors (base_color, ColorDefs[MT_COLOR_INDICATOR]);
    }
  }
  else
  {
    tree_color = ColorDefs[MT_COLOR_TREE];
    if (use_indicator)
    {
      tree_color = ColorDefs[MT_COLOR_INDICATOR];
      base_color = ColorDefs[MT_COLOR_INDICATOR];
    }
  }

  ATTRSET (base_color);

  memset (&mbstate, 0, sizeof (mbstate));
  while (*s)
  {
    if (*s < MUTT_TREE_MAX)
    {
      ATTRSET (tree_color);
      while (*s && *s < MUTT_TREE_MAX)
      {
	switch (*s)
	{
	  case MUTT_TREE_LLCORNER:
	    if (option (OPTASCIICHARS))
	      addch ('`');
#ifdef WACS_LLCORNER
	    else
	      add_wch(WACS_LLCORNER);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\224"); /* WACS_LLCORNER */
	    else
	      addch (ACS_LLCORNER);
#endif
	    break;
	  case MUTT_TREE_ULCORNER:
	    if (option (OPTASCIICHARS))
	      addch (',');
#ifdef WACS_ULCORNER
	    else
	      add_wch(WACS_ULCORNER);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\214"); /* WACS_ULCORNER */
	    else
	      addch (ACS_ULCORNER);
#endif
	    break;
	  case MUTT_TREE_LTEE:
	    if (option (OPTASCIICHARS))
	      addch ('|');
#ifdef WACS_LTEE
	    else
	      add_wch(WACS_LTEE);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\234"); /* WACS_LTEE */
	    else
	      addch (ACS_LTEE);
#endif
	    break;
	  case MUTT_TREE_HLINE:
	    if (option (OPTASCIICHARS))
	      addch ('-');
#ifdef WACS_HLINE
	    else
	      add_wch(WACS_HLINE);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\200"); /* WACS_HLINE */
	    else
	      addch (ACS_HLINE);
#endif
	    break;
	  case MUTT_TREE_VLINE:
	    if (option (OPTASCIICHARS))
	      addch ('|');
#ifdef WACS_VLINE
	    else
	      add_wch(WACS_VLINE);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\202"); /* WACS_VLINE */
	    else
	      addch (ACS_VLINE);
#endif
	    break;
	  case MUTT_TREE_TTEE:
	    if (option (OPTASCIICHARS))
	      addch ('-');
#ifdef WACS_TTEE
	    else
	      add_wch(WACS_TTEE);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\254"); /* WACS_TTEE */
	    else
	      addch (ACS_TTEE);
#endif
	    break;
	  case MUTT_TREE_BTEE:
	    if (option (OPTASCIICHARS))
	      addch ('-');
#ifdef WACS_BTEE
	    else
	      add_wch(WACS_BTEE);
#else
	    else if (Charset_is_utf8)
	      addstr ("\342\224\264"); /* WACS_BTEE */
	    else
	      addch (ACS_BTEE);
#endif
	    break;
	  case MUTT_TREE_SPACE:
	    addch (' ');
	    break;
	  case MUTT_TREE_RARROW:
	    addch ('>');
	    break;
	  case MUTT_TREE_STAR:
	    addch ('*'); /* fake thread indicator */
	    break;
	  case MUTT_TREE_HIDDEN:
	    addch ('&');
	    break;
	  case MUTT_TREE_EQUALS:
	    addch ('=');
	    break;
	  case MUTT_TREE_MISSING:
	    addch ('?');
	    break;
	}
	s++, n--;
      }
      ATTRSET(base_color);
    }
    else if ((k = mbrtowc (&wc, (char *)s, n, &mbstate)) > 0)
    {
      addnstr ((char *)s, k);
      s += k, n-= k;
    }
    else
      break;
  }
}

static void menu_make_entry (char *s, int l, MUTTMENU *menu, int i)
{
  if (menu->dialog)
  {
    strncpy (s, NONULL (menu->dialog[i]), l);
    menu->current = -1; /* hide menubar */
  }
  else
    menu->make_entry (s, l, menu, i);
}

static void menu_pad_string (MUTTMENU *menu, char *s, size_t n)
{
  char *scratch = safe_strdup (s);
  int shift = option (OPTARROWCURSOR) ? 3 : 0;
  int cols = menu->indexwin->cols - shift;

  mutt_format_string (s, n, cols, cols, FMT_LEFT, ' ', scratch, mutt_strlen (scratch), 1);
  s[n - 1] = 0;
  FREE (&scratch);
}

void menu_redraw_full (MUTTMENU *menu)
{
#if ! (defined (USE_SLANG_CURSES) || defined (HAVE_RESIZETERM))
  mutt_reflow_windows ();
#endif
  NORMAL_COLOR;
  /* clear() doesn't optimize screen redraws */
  move (0, 0);
  clrtobot ();

  if (option (OPTHELP))
  {
    SETCOLOR (MT_COLOR_STATUS);
    mutt_window_move (menu->helpwin, 0, 0);
    mutt_paddstr (menu->helpwin->cols, menu->help);
    NORMAL_COLOR;
  }
  menu->offset = 0;
  menu->pagelen = menu->indexwin->rows;

  mutt_show_error ();

  menu->redraw = REDRAW_INDEX | REDRAW_STATUS;
#ifdef USE_SIDEBAR
  menu->redraw |= REDRAW_SIDEBAR;
#endif
}

void menu_redraw_status (MUTTMENU *menu)
{
  char buf[STRING];

  snprintf (buf, sizeof (buf), MUTT_MODEFMT, menu->title);
  SETCOLOR (MT_COLOR_STATUS);
  mutt_window_move (menu->statuswin, 0, 0);
  mutt_paddstr (menu->statuswin->cols, buf);
  NORMAL_COLOR;
  menu->redraw &= ~REDRAW_STATUS;
}

#ifdef USE_SIDEBAR
void menu_redraw_sidebar (MUTTMENU *menu)
{
  menu->redraw &= ~REDRAW_SIDEBAR;
  mutt_sb_draw ();
}
#endif

void menu_redraw_index (MUTTMENU *menu)
{
  char buf[LONG_STRING];
  int i;
  COLOR_ATTR attr;

  for (i = menu->top; i < menu->top + menu->pagelen; i++)
  {
    if (i < menu->max)
    {
      attr = menu->color(i);

      menu_make_entry (buf, sizeof (buf), menu, i);
      menu_pad_string (menu, buf, sizeof (buf));
      mutt_window_move (menu->indexwin, i - menu->top + menu->offset, 0);

      if (i == menu->current)
      {
        if (option(OPTARROWCURSOR))
        {
          SETCOLOR(MT_COLOR_INDICATOR);
          addstr ("->");
          ATTRSET(attr);
          addch (' ');
          print_enriched_string (attr, (unsigned char *) buf, 0);
        }
        else
          print_enriched_string (attr, (unsigned char *) buf, 1);
      }
      else
      {
        if (option(OPTARROWCURSOR))
        {
          ATTRSET(attr);
          addstr("   ");
        }
        print_enriched_string (attr, (unsigned char *) buf, 0);
      }
    }
    else
    {
      NORMAL_COLOR;
      mutt_window_clearline (menu->indexwin, i - menu->top + menu->offset);
    }
  }
  NORMAL_COLOR;
  menu->redraw = 0;
}

void menu_redraw_motion (MUTTMENU *menu)
{
  char buf[LONG_STRING];
  COLOR_ATTR old_color, cur_color;

  if (menu->dialog)
  {
    menu->redraw &= ~REDRAW_MOTION;
    return;
  }

  /* Note: menu->color() for the index can end up retrieving a message
   * over imap (if matching against ~h for instance).  This can
   * generate status messages.  So we want to call it *before* we
   * position the cursor for drawing. */
  old_color = menu->color (menu->oldcurrent);
  mutt_window_move (menu->indexwin, menu->oldcurrent + menu->offset - menu->top, 0);

  if (option (OPTARROWCURSOR))
  {
    /* clear the pointer */
    ATTRSET(old_color);
    addstr ("  ");

    if (menu->redraw & REDRAW_MOTION_RESYNCH)
    {
      menu_make_entry (buf, sizeof (buf), menu, menu->oldcurrent);
      menu_pad_string (menu, buf, sizeof (buf));
      mutt_window_move (menu->indexwin, menu->oldcurrent + menu->offset - menu->top, 3);
      print_enriched_string (old_color, (unsigned char *) buf, 0);
    }

    /* now draw it in the new location */
    SETCOLOR(MT_COLOR_INDICATOR);
    mutt_window_mvaddstr (menu->indexwin, menu->current + menu->offset - menu->top, 0, "->");
  }
  else
  {
    /* erase the current indicator */
    menu_make_entry (buf, sizeof (buf), menu, menu->oldcurrent);
    menu_pad_string (menu, buf, sizeof (buf));
    print_enriched_string (old_color, (unsigned char *) buf, 0);

    /* now draw the new one to reflect the change */
    cur_color = menu->color (menu->current);
    menu_make_entry (buf, sizeof (buf), menu, menu->current);
    menu_pad_string (menu, buf, sizeof (buf));
    mutt_window_move (menu->indexwin, menu->current + menu->offset - menu->top, 0);
    print_enriched_string (cur_color, (unsigned char *) buf, 1);
  }
  menu->redraw &= REDRAW_STATUS;
  NORMAL_COLOR;
}

void menu_redraw_current (MUTTMENU *menu)
{
  char buf[LONG_STRING];
  COLOR_ATTR attr = menu->color (menu->current);

  mutt_window_move (menu->indexwin, menu->current + menu->offset - menu->top, 0);
  menu_make_entry (buf, sizeof (buf), menu, menu->current);
  menu_pad_string (menu, buf, sizeof (buf));

  if (option (OPTARROWCURSOR))
  {
    SETCOLOR(MT_COLOR_INDICATOR);
    addstr ("->");
    ATTRSET(attr);
    addch (' ');
    menu_pad_string (menu, buf, sizeof (buf));
    print_enriched_string (attr, (unsigned char *) buf, 0);
  }
  else
    print_enriched_string (attr, (unsigned char *) buf, 1);

  menu->redraw &= REDRAW_STATUS;
  NORMAL_COLOR;
}

static void menu_redraw_prompt (MUTTMENU *menu)
{
  if (menu->dialog)
  {
    if (option (OPTMSGERR))
    {
      mutt_sleep (1);
      unset_option (OPTMSGERR);
    }

    if (*Errorbuf)
      mutt_clear_error ();

    mutt_window_mvaddstr (menu->messagewin, 0, 0, menu->prompt);
    mutt_window_clrtoeol (menu->messagewin);
  }
}

void menu_check_recenter (MUTTMENU *menu)
{
  int c = MIN (MenuContext, menu->pagelen / 2);
  int old_top = menu->top;

  if (!option (OPTMENUMOVEOFF) && menu->max <= menu->pagelen) /* less entries than lines */
  {
    if (menu->top != 0)
    {
      menu->top = 0;
      menu->redraw |= REDRAW_INDEX;
    }
  }
  else
  {
    if (option (OPTMENUSCROLL) || (menu->pagelen <= 0) || (c < MenuContext))
    {
      if (menu->current < menu->top + c)
	menu->top = menu->current - c;
      else if (menu->current >= menu->top + menu->pagelen - c)
	menu->top = menu->current - menu->pagelen + c + 1;
    }
    else
    {
      if (menu->current < menu->top + c)
	menu->top -= (menu->pagelen - c) * ((menu->top + menu->pagelen - 1 - menu->current) / (menu->pagelen - c)) - c;
      else if ((menu->current >= menu->top + menu->pagelen - c))
	menu->top += (menu->pagelen - c) * ((menu->current - menu->top) / (menu->pagelen - c)) - c;
    }
  }

  if (!option (OPTMENUMOVEOFF)) /* make entries stick to bottom */
    menu->top = MIN (menu->top, menu->max - menu->pagelen);
  menu->top = MAX (menu->top, 0);

  if (menu->top != old_top)
    menu->redraw |= REDRAW_INDEX;
}

void menu_jump (MUTTMENU *menu)
{
  int n;
  char buf[SHORT_STRING];

  if (menu->max)
  {
    mutt_unget_event (LastKey, 0);
    buf[0] = 0;
    if (mutt_get_field (_("Jump to: "), buf, sizeof (buf), 0) == 0 && buf[0])
    {
      if (mutt_atoi (buf, &n, 0) == 0 && n > 0 && n < menu->max + 1)
      {
	n--;	/* msg numbers are 0-based */
	menu->current = n;
	menu->redraw = REDRAW_MOTION;
      }
      else
	mutt_error _("Invalid index number.");
    }
  }
  else
    mutt_error _("No entries.");
}

void menu_next_line (MUTTMENU *menu)
{
  if (menu->max)
  {
    int c = MIN (MenuContext, menu->pagelen / 2);

    if ((menu->top + 1 < menu->max - c) &&
        (option(OPTMENUMOVEOFF) ||
         (menu->max > menu->pagelen && menu->top < menu->max - menu->pagelen)))
    {
      menu->top++;
      if (menu->current < menu->top + c && menu->current < menu->max - 1)
	menu->current++;
      menu->redraw = REDRAW_INDEX;
    }
    else
      mutt_error _("You cannot scroll down farther.");
  }
  else
    mutt_error _("No entries.");
}

void menu_prev_line (MUTTMENU *menu)
{
  if (menu->top > 0)
  {
    int c = MIN (MenuContext, menu->pagelen / 2);

    menu->top--;
    if (menu->current >= menu->top + menu->pagelen - c && menu->current > 1)
      menu->current--;
    menu->redraw = REDRAW_INDEX;
  }
  else
    mutt_error _("You cannot scroll up farther.");
}

/*
 * pageup:   jumplen == -pagelen
 * pagedown: jumplen == pagelen
 * halfup:   jumplen == -pagelen/2
 * halfdown: jumplen == pagelen/2
 */
#define DIRECTION ((neg * 2) + 1)
static void menu_length_jump (MUTTMENU *menu, int jumplen)
{
  int tmp, neg = (jumplen >= 0) ? 0 : -1;
  int c = MIN (MenuContext, menu->pagelen / 2);

  if (menu->max)
  {
    /* possible to scroll? */
    if (DIRECTION * menu->top <
	(tmp = (neg ? 0 : (menu->max /*-1*/) - (menu->pagelen /*-1*/))))
    {
      menu->top += jumplen;

      /* jumped too long? */
      if ((neg || !option (OPTMENUMOVEOFF)) &&
	  DIRECTION * menu->top > tmp)
	menu->top = tmp;

      /* need to move the cursor? */
      if ((DIRECTION *
	   (tmp = (menu->current -
		   (menu->top + (neg ? (menu->pagelen - 1) - c : c))
             ))) < 0)
	menu->current -= tmp;

      menu->redraw = REDRAW_INDEX;
    }
    else if (menu->current != (neg ? 0 : menu->max - 1) && !menu->dialog)
    {
      menu->current += jumplen;
      menu->redraw = REDRAW_MOTION;
    }
    else
      mutt_error (neg ? _("You are on the first page.")
                  : _("You are on the last page."));

    menu->current = MIN (menu->current, menu->max - 1);
    menu->current = MAX (menu->current, 0);
  }
  else
    mutt_error _("No entries.");
}
#undef DIRECTION

void menu_next_page (MUTTMENU *menu)
{
  menu_length_jump (menu, MAX (menu->pagelen /* - MenuOverlap */, 0));
}

void menu_prev_page (MUTTMENU *menu)
{
  menu_length_jump (menu, 0 - MAX (menu->pagelen /* - MenuOverlap */, 0));
}

void menu_half_down (MUTTMENU *menu)
{
  menu_length_jump (menu, menu->pagelen / 2);
}

void menu_half_up (MUTTMENU *menu)
{
  menu_length_jump (menu, 0 - menu->pagelen / 2);
}

void menu_top_page (MUTTMENU *menu)
{
  if (menu->current != menu->top)
  {
    menu->current = menu->top;
    menu->redraw = REDRAW_MOTION;
  }
}

void menu_bottom_page (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->current = menu->top + menu->pagelen - 1;
    if (menu->current > menu->max - 1)
      menu->current = menu->max - 1;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("No entries.");
}

void menu_middle_page (MUTTMENU *menu)
{
  int i;

  if (menu->max)
  {
    i = menu->top + menu->pagelen;
    if (i > menu->max - 1)
      i = menu->max - 1;
    menu->current = menu->top + (i - menu->top) / 2;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("No entries.");
}

void menu_first_entry (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->current = 0;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("No entries.");
}

void menu_last_entry (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->current = menu->max - 1;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("No entries.");
}

void menu_current_top (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->top = menu->current;
    menu->redraw = REDRAW_INDEX;
  }
  else
    mutt_error _("No entries.");
}

void menu_current_middle (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->top = menu->current - menu->pagelen / 2;
    if (menu->top < 0)
      menu->top = 0;
    menu->redraw = REDRAW_INDEX;
  }
  else
    mutt_error _("No entries.");
}

void menu_current_bottom (MUTTMENU *menu)
{
  if (menu->max)
  {
    menu->top = menu->current - menu->pagelen + 1;
    if (menu->top < 0)
      menu->top = 0;
    menu->redraw = REDRAW_INDEX;
  }
  else
    mutt_error _("No entries.");
}

static void menu_next_entry (MUTTMENU *menu)
{
  if (menu->current < menu->max - 1)
  {
    menu->current++;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("You are on the last entry.");
}

static void menu_prev_entry (MUTTMENU *menu)
{
  if (menu->current)
  {
    menu->current--;
    menu->redraw = REDRAW_MOTION;
  }
  else
    mutt_error _("You are on the first entry.");
}

static COLOR_ATTR default_color (int i)
{
  return ColorDefs[MT_COLOR_NORMAL];
}

static int menu_search_generic (MUTTMENU *m, regex_t *re, int n)
{
  char buf[LONG_STRING];

  menu_make_entry (buf, sizeof (buf), m, n);
  return (regexec (re, buf, 0, NULL, 0));
}

void mutt_menu_init (void)
{
  int i;

  for (i = 0; i < MENU_MAX; i++)
    SearchBuffers[i] = NULL;
}

MUTTMENU *mutt_new_menu (int menu)
{
  MUTTMENU *p = (MUTTMENU *) safe_calloc (1, sizeof (MUTTMENU));

  if ((menu < 0) || (menu >= MENU_MAX))
    menu = MENU_GENERIC;

  p->menu = menu;
  p->current = 0;
  p->top = 0;
  p->offset = 0;
  p->redraw = REDRAW_FULL;
  p->pagelen = MuttIndexWindow->rows;
  p->indexwin = MuttIndexWindow;
  p->statuswin = MuttStatusWindow;
  p->helpwin = MuttHelpWindow;
  p->messagewin = MuttMessageWindow;
  p->color = default_color;
  p->search = menu_search_generic;

  return (p);
}

void mutt_menuDestroy (MUTTMENU **p)
{
  int i;

  if ((*p)->dialog)
  {
    for (i=0; i < (*p)->max; i++)
      FREE (&(*p)->dialog[i]);

    FREE (& (*p)->dialog);
  }

  FREE (p);		/* __FREE_CHECKED__ */
}

void mutt_menu_add_dialog_row (MUTTMENU *m, const char *row)
{
  if (m->dsize <= m->max)
  {
    m->dsize += 10;
    safe_realloc (&m->dialog, m->dsize * sizeof (char *));
  }
  m->dialog[m->max++] = safe_strdup (row);
}

static MUTTMENU *get_current_menu (void)
{
  return MenuStackCount ? MenuStack[MenuStackCount - 1] : NULL;
}

void mutt_push_current_menu (MUTTMENU *menu)
{
  if (MenuStackCount >= MenuStackLen)
  {
    MenuStackLen += 5;
    safe_realloc (&MenuStack, MenuStackLen * sizeof(MUTTMENU *));
  }

  MenuStack[MenuStackCount++] = menu;
  CurrentMenu = menu->menu;
}

void mutt_pop_current_menu (MUTTMENU *menu)
{
  MUTTMENU *prev_menu;

  if (!MenuStackCount ||
      (MenuStack[MenuStackCount - 1] != menu))
  {
    dprint (1, (debugfile, "mutt_pop_current_menu() called with inactive menu\n"));
    return;
  }

  MenuStackCount--;
  prev_menu = get_current_menu ();
  if (prev_menu)
  {
    CurrentMenu = prev_menu->menu;
    /* REDRAW_FLOW is for the pager, which needs to reflow if
     * a window resize or setting change occurred.
     */
    prev_menu->redraw = REDRAW_FULL | REDRAW_FLOW;
  }
  else
  {
    CurrentMenu = MENU_MAIN;
    /* Clearing when Mutt exits would be an annoying change in
     * behavior for those who have disabled alternative screens.  The
     * option is currently set by autocrypt initialization which mixes
     * menus and prompts outside of the normal menu system state.
     */
    if (option (OPTMENUPOPCLEARSCREEN))
    {
      move (0, 0);
      clrtobot ();
    }
  }
}

void mutt_set_current_menu_redraw (int redraw)
{
  MUTTMENU *current_menu;

  current_menu = get_current_menu ();
  if (current_menu)
    current_menu->redraw |= redraw;
}

void mutt_set_current_menu_redraw_full (void)
{
  MUTTMENU *current_menu;

  current_menu = get_current_menu ();
  if (current_menu)
    current_menu->redraw = REDRAW_FULL;
}

void mutt_set_menu_redraw (int menu_type, int redraw)
{
  if (CurrentMenu == menu_type)
    mutt_set_current_menu_redraw (redraw);
}

void mutt_set_menu_redraw_full (int menu_type)
{
  if (CurrentMenu == menu_type)
    mutt_set_current_menu_redraw_full ();
}

void mutt_current_menu_redraw (void)
{
  MUTTMENU *current_menu;

  current_menu = get_current_menu ();
  if (current_menu)
  {
    if (menu_redraw (current_menu) == OP_REDRAW)
      /* On a REDRAW_FULL with a non-customized redraw, menu_redraw()
       * will return OP_REDRAW to give the calling menu-loop a chance to
       * customize output.
       */
      menu_redraw (current_menu);
  }
}


#define MUTT_SEARCH_UP   1
#define MUTT_SEARCH_DOWN 2

static int menu_search (MUTTMENU *menu, int op)
{
  int r, wrap = 0;
  int searchDir;
  regex_t re;
  char buf[SHORT_STRING];
  char* searchBuf = menu->menu >= 0 && menu->menu < MENU_MAX ?
    SearchBuffers[menu->menu] : NULL;

  if (!(searchBuf && *searchBuf) ||
      (op != OP_SEARCH_NEXT && op != OP_SEARCH_OPPOSITE))
  {
    strfcpy (buf, searchBuf && *searchBuf ? searchBuf : "", sizeof (buf));
    if (mutt_get_field ((op == OP_SEARCH || op == OP_SEARCH_NEXT)
			? _("Search for: ") : _("Reverse search for: "),
			buf, sizeof (buf), MUTT_CLEAR) != 0 || !buf[0])
      return (-1);
    if (menu->menu >= 0 && menu->menu < MENU_MAX)
    {
      mutt_str_replace (&SearchBuffers[menu->menu], buf);
      searchBuf = SearchBuffers[menu->menu];
    }
    menu->searchDir = (op == OP_SEARCH || op == OP_SEARCH_NEXT) ?
      MUTT_SEARCH_DOWN : MUTT_SEARCH_UP;
  }

  searchDir = (menu->searchDir == MUTT_SEARCH_UP) ? -1 : 1;
  if (op == OP_SEARCH_OPPOSITE)
    searchDir = -searchDir;

  if ((r = REGCOMP (&re, searchBuf, REG_NOSUB | mutt_which_case (searchBuf))) != 0)
  {
    regerror (r, &re, buf, sizeof (buf));
    mutt_error ("%s", buf);
    return (-1);
  }

  r = menu->current + searchDir;
search_next:
  if (wrap)
    mutt_message (_("Search wrapped to top."));
  while (r >= 0 && r < menu->max)
  {
    if (menu->search (menu, &re, r) == 0)
    {
      regfree (&re);
      return r;
    }

    r += searchDir;
  }

  if (option (OPTWRAPSEARCH) && wrap++ == 0)
  {
    r = searchDir == 1 ? 0 : menu->max - 1;
    goto search_next;
  }
  regfree (&re);
  mutt_error _("Not found.");
  return (-1);
}

static int menu_dialog_translate_op (int i)
{
  switch (i)
  {
    case OP_NEXT_ENTRY:
      return OP_NEXT_LINE;
    case OP_PREV_ENTRY:
      return OP_PREV_LINE;
    case OP_CURRENT_TOP:   case OP_FIRST_ENTRY:
      return OP_TOP_PAGE;
    case OP_CURRENT_BOTTOM:    case OP_LAST_ENTRY:
      return OP_BOTTOM_PAGE;
    case OP_CURRENT_MIDDLE:
      return OP_MIDDLE_PAGE;
  }

  return i;
}

static int menu_dialog_dokey (MUTTMENU *menu, int *ip)
{
  event_t ch;
  char *p;

  do
  {
    ch = mutt_getch();
  } while (ch.ch == -2);

  if (ch.ch < 0)
  {
    *ip = -1;
    return 0;
  }

  if (ch.ch && (p = strchr (menu->keys, ch.ch)))
  {
    *ip = OP_MAX + (p - menu->keys + 1);
    return 0;
  }
  else
  {
    mutt_unget_event (ch.op ? 0 : ch.ch, ch.op ? ch.op : 0);
    return -1;
  }
}

int menu_redraw (MUTTMENU *menu)
{
  if (menu->custom_menu_redraw)
  {
    menu->custom_menu_redraw (menu);
    return OP_NULL;
  }

  /* See if all or part of the screen needs to be updated.  */
  if (menu->redraw & REDRAW_FULL)
  {
    menu_redraw_full (menu);
    /* allow the caller to do any local configuration */
    return (OP_REDRAW);
  }

  if (!menu->dialog)
    menu_check_recenter (menu);

  if (menu->redraw & REDRAW_STATUS)
    menu_redraw_status (menu);
#ifdef USE_SIDEBAR
  if (menu->redraw & REDRAW_SIDEBAR)
    menu_redraw_sidebar (menu);
#endif
  if (menu->redraw & REDRAW_INDEX)
    menu_redraw_index (menu);
  else if (menu->redraw & (REDRAW_MOTION | REDRAW_MOTION_RESYNCH))
    menu_redraw_motion (menu);
  else if (menu->redraw == REDRAW_CURRENT)
    menu_redraw_current (menu);

  if (menu->dialog)
    menu_redraw_prompt (menu);

  return OP_NULL;
}

int mutt_menuLoop (MUTTMENU *menu)
{
  int i = OP_NULL;

  FOREVER
  {
    if (option (OPTMENUCALLER))
    {
      unset_option (OPTMENUCALLER);
      return OP_NULL;
    }

    /* Clear the tag prefix unless we just started it.  Don't clear
     * the prefix on a timeout (i==-2), but do clear on an abort (i==-1)
     */
    if (menu->tagprefix &&
        i != OP_TAG_PREFIX && i != OP_TAG_PREFIX_COND && i != -2)
      menu->tagprefix = 0;

    mutt_curs_set (0);

#if defined (USE_SLANG_CURSES) || defined (HAVE_RESIZETERM)
    if (SigWinch)
    {
      do
      {
        SigWinch = 0;
        mutt_resize_screen ();
      }
      while (SigWinch);

      clearok(stdscr,TRUE);/*force complete redraw*/
    }
#endif

    if (menu->custom_menu_update)
      menu->custom_menu_update (menu);

    if (menu_redraw (menu) == OP_REDRAW)
      return OP_REDRAW;

    /* give visual indication that the next command is a tag- command */
    if (menu->tagprefix)
    {
      mutt_window_mvaddstr (menu->messagewin, 0, 0, "tag-");
      mutt_window_clrtoeol (menu->messagewin);
    }

    menu->oldcurrent = menu->current;


    /* move the cursor out of the way */
    if (option (OPTARROWCURSOR))
      mutt_window_move (menu->indexwin, menu->current - menu->top + menu->offset, 2);
    else if (option (OPTBRAILLEFRIENDLY))
      mutt_window_move (menu->indexwin, menu->current - menu->top + menu->offset, 0);
    else
      mutt_window_move (menu->indexwin, menu->current - menu->top + menu->offset,
                        menu->indexwin->cols - 1);

    mutt_refresh ();

    /* try to catch dialog keys before ops */
    if (menu->dialog && menu_dialog_dokey (menu, &i) == 0)
      return i;

    i = km_dokey (menu->menu);
    if (i == OP_TAG_PREFIX || i == OP_TAG_PREFIX_COND)
    {
      if (menu->tagprefix)
      {
        menu->tagprefix = 0;
        mutt_window_clearline (menu->messagewin, 0);
        continue;
      }

      if (menu->tagged)
      {
	menu->tagprefix = 1;
        continue;
      }
      else if (i == OP_TAG_PREFIX)
      {
	mutt_error _("No tagged entries.");
	i = -1;
      }
      else /* None tagged, OP_TAG_PREFIX_COND */
      {
	mutt_flush_macro_to_endcond ();
	mutt_message _("Nothing to do.");
	i = -1;
      }
    }
    else if (menu->tagged && option (OPTAUTOTAG))
      menu->tagprefix = 1;

    mutt_curs_set (1);

    if (i < 0)
    {
      if (menu->tagprefix)
        mutt_window_clearline (menu->messagewin, 0);
      continue;
    }

    if (!menu->dialog)
      mutt_clear_error ();

    /* Convert menubar movement to scrolling */
    if (menu->dialog)
      i = menu_dialog_translate_op (i);

    switch (i)
    {
      case OP_NEXT_ENTRY:
	menu_next_entry (menu);
	break;
      case OP_PREV_ENTRY:
	menu_prev_entry (menu);
	break;
      case OP_HALF_DOWN:
	menu_half_down (menu);
	break;
      case OP_HALF_UP:
	menu_half_up (menu);
	break;
      case OP_NEXT_PAGE:
	menu_next_page (menu);
	break;
      case OP_PREV_PAGE:
	menu_prev_page (menu);
	break;
      case OP_NEXT_LINE:
	menu_next_line (menu);
	break;
      case OP_PREV_LINE:
	menu_prev_line (menu);
	break;
      case OP_FIRST_ENTRY:
	menu_first_entry (menu);
	break;
      case OP_LAST_ENTRY:
	menu_last_entry (menu);
	break;
      case OP_TOP_PAGE:
	menu_top_page (menu);
	break;
      case OP_MIDDLE_PAGE:
	menu_middle_page (menu);
	break;
      case OP_BOTTOM_PAGE:
	menu_bottom_page (menu);
	break;
      case OP_CURRENT_TOP:
	menu_current_top (menu);
	break;
      case OP_CURRENT_MIDDLE:
	menu_current_middle (menu);
	break;
      case OP_CURRENT_BOTTOM:
	menu_current_bottom (menu);
	break;
      case OP_SEARCH:
      case OP_SEARCH_REVERSE:
      case OP_SEARCH_NEXT:
      case OP_SEARCH_OPPOSITE:
	if (menu->search && !menu->dialog) /* Searching dialogs won't work */
	{
	  menu->oldcurrent = menu->current;
	  if ((menu->current = menu_search (menu, i)) != -1)
	    menu->redraw = REDRAW_MOTION;
	  else
	    menu->current = menu->oldcurrent;
	}
	else
	  mutt_error _("Search is not implemented for this menu.");
	break;

      case OP_JUMP:
	if (menu->dialog)
	  mutt_error _("Jumping is not implemented for dialogs.");
	else
	  menu_jump (menu);
	break;

      case OP_ENTER_COMMAND:
	mutt_enter_command ();
	break;

      case OP_TAG:
	if (menu->tag && !menu->dialog)
	{
	  if (menu->tagprefix && !option (OPTAUTOTAG))
	  {
	    for (i = 0; i < menu->max; i++)
	      menu->tagged += menu->tag (menu, i, 0);
	    menu->redraw |= REDRAW_INDEX;
	  }
	  else if (menu->max)
	  {
	    int i = menu->tag (menu, menu->current, -1);
	    menu->tagged += i;
	    if (i && option (OPTRESOLVE) && menu->current < menu->max - 1)
	    {
	      menu->current++;
	      menu->redraw |= REDRAW_MOTION_RESYNCH;
	    }
	    else
	      menu->redraw |= REDRAW_CURRENT;
	  }
	  else
	    mutt_error _("No entries.");
	}
	else
	  mutt_error _("Tagging is not supported.");
	break;

      case OP_SHELL_ESCAPE:
	mutt_shell_escape ();
	break;

      case OP_WHAT_KEY:
	mutt_what_key ();
	break;

      case OP_CHECK_STATS:
	mutt_check_stats ();
	break;

      case OP_REDRAW:
	clearok (stdscr, TRUE);
	menu->redraw = REDRAW_FULL;
	break;

      case OP_HELP:
	mutt_help (menu->menu);
	menu->redraw = REDRAW_FULL;
	break;

      case OP_ERROR_HISTORY:
	mutt_error_history_display ();
	menu->redraw = REDRAW_FULL;
	break;

      case OP_NULL:
	km_error_key (menu->menu);
	break;

      case OP_END_COND:
	break;

      default:
	return (i);
    }
  }
  /* not reached */
}