Code

unicode fixes from Dmitry Baryshkov/René van Bevern
[ncmpc.git] / src / list_window.c
index 6b4a41f0c7591290a487f3b3c0af1ebf11d0e23f..05f5630e9b4e5d6228132a16edcd736a17c71278 100644 (file)
@@ -1,5 +1,7 @@
 /* 
- * (c) 2004 by Kalle Wallin (kaw@linux.se)
+ * $Id$
+ *
+ * (c) 2004 by Kalle Wallin <kaw@linux.se>
  *
  * 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
 #include "colors.h"
 #include "list_window.h"
 
+extern void screen_bell(void);
+
 list_window_t *
 list_window_init(WINDOW *w, int width, int height)
 {
   list_window_t *lw;
 
-  lw = g_malloc(sizeof(list_window_t));
-  memset(lw, 0, sizeof(list_window_t));
+  lw = g_malloc0(sizeof(list_window_t));
   lw->w = w;
   lw->cols = width;
   lw->rows = height;
@@ -58,6 +61,7 @@ void
 list_window_reset(list_window_t *lw)
 {
   lw->selected = 0;
+  lw->xoffset = 0;
   lw->start = 0;
   lw->clear = 1;
 }
@@ -89,24 +93,43 @@ list_window_next(list_window_t *lw, int length)
 {
   if( lw->selected < length-1 )
     lw->selected++;
+  else if ( options.list_wrap )
+    lw->selected = 0;
 }
 
 void
-list_window_previous(list_window_t *lw)
+list_window_previous(list_window_t *lw, int length)
 {
   if( lw->selected > 0 )
     lw->selected--;
+  else if( options.list_wrap )
+    lw->selected = length-1;
+}
+
+void
+list_window_right(list_window_t *lw, int length)
+{
+  lw->xoffset++;
+}
+
+void
+list_window_left(list_window_t *lw)
+{
+  if( lw->xoffset > 0 )
+    lw->xoffset--;
 }
 
 void
 list_window_first(list_window_t *lw)
 {
+  lw->xoffset = 0;
   lw->selected = 0;
 }
 
 void
 list_window_last(list_window_t *lw, int length)
 {
+  lw->xoffset = 0;
   lw->selected = length-1;
 }
 
@@ -142,18 +165,22 @@ list_window_paint(list_window_t *lw,
 {
   int i;
   int fill = options.wide_cursor;
+  int show_cursor = !(lw->flags & LW_HIDE_CURSOR);
 
-  while( lw->selected < lw->start )
+  if( show_cursor )
     {
-      lw->start--;
-      lw->clear=1;
-    }
-  while( lw->selected >= lw->start+lw->rows )
-    {
-      lw->start++;
-      lw->clear=1;
+      while( lw->selected < lw->start )
+       {
+         lw->start--;
+         lw->clear=1;
+       }
+      while( lw->selected >= lw->start+lw->rows )
+       {
+         lw->start++;
+         lw->clear=1;
+       }
     }
-
+  
   for(i=0; i<lw->rows; i++)
     {
       int highlight = 0;
@@ -166,18 +193,20 @@ list_window_paint(list_window_t *lw,
       if( label )
        {
          int selected = lw->start+i == lw->selected;
+         size_t len = my_strlen(label);
 
          if( highlight )
            colors_use(lw->w, COLOR_LIST_BOLD);
          else
            colors_use(lw->w, COLOR_LIST);
 
-         if( selected )
+         if( show_cursor && selected )
            wattron(lw->w, A_REVERSE);
          
-         waddnstr(lw->w, label, lw->cols-1);
-         if( fill )
-           mvwhline(lw->w, i, strlen(label), ' ', lw->cols-1);
+         //waddnstr(lw->w, label, lw->cols);
+         waddnstr(lw->w, label);
+         if( fill && len<lw->cols )
+           mvwhline(lw->w, i, len, ' ', lw->cols-len);
 
          if( selected )
            wattroff(lw->w, A_REVERSE);   
@@ -214,8 +243,10 @@ list_window_find(list_window_t *lw,
        }
       if( wrap )
        {
+         if ( i==0 ) /* empty list */
+           return 1;
          i=0; /* first item */
-         beep(); 
+         screen_bell();
        }
     }
   return 1;
@@ -234,6 +265,9 @@ list_window_rfind(list_window_t *lw,
   int i = lw->selected-1;
   char *label;
 
+  if ( rows == 0 )
+    return 1;
+
   while( wrap || i==lw->selected-1 )
     {
       while( i>=0 && (label=(callback) (i,&h,callback_data)) )
@@ -250,7 +284,7 @@ list_window_rfind(list_window_t *lw,
       if( wrap )
        {
          i=rows-1; /* last item */
-         beep();
+         screen_bell();
        }
     }
   return 1;
@@ -264,7 +298,7 @@ list_window_cmd(list_window_t *lw, int rows, command_t cmd)
   switch(cmd)
     {
     case CMD_LIST_PREVIOUS:
-      list_window_previous(lw);
+      list_window_previous(lw, rows);
       lw->repaint=1;
       break;
     case CMD_LIST_NEXT:
@@ -294,3 +328,62 @@ list_window_cmd(list_window_t *lw, int rows, command_t cmd)
 }
 
 
+
+
+
+list_window_state_t *
+list_window_init_state(void)
+{
+  return g_malloc0(sizeof(list_window_state_t));
+}
+
+list_window_state_t *
+list_window_free_state(list_window_state_t *state)
+{
+  if( state )
+    {
+      if( state->list )
+       {
+         GList *list = state->list;
+         while( list )
+           {
+             g_free(list->data);
+             list->data = NULL;
+             list = list->next;
+           }
+         g_list_free(state->list);
+         state->list = NULL;
+       }
+      g_free(state);
+    }
+  return NULL;
+}
+
+void 
+list_window_push_state(list_window_state_t *state, list_window_t *lw)
+{
+  if( state )
+    {
+      list_window_t *tmp = g_malloc(sizeof(list_window_t));
+      memcpy(tmp, lw, sizeof(list_window_t));
+      state->list = g_list_prepend(state->list, (gpointer) tmp);
+      list_window_reset(lw);
+    }
+}
+
+void 
+list_window_pop_state(list_window_state_t *state, list_window_t *lw)
+{
+  if( state && state->list )
+    {
+      list_window_t *tmp = state->list->data;
+
+      memcpy(lw, tmp, sizeof(list_window_t));
+      g_free(tmp);
+      state->list->data = NULL;
+      state->list = g_list_delete_link(state->list, state->list);
+    }
+}
+
+
+