]> pd.if.org Git - pd_readline/commitdiff
More good progress.
authorandy <andy@obsidian.(none)>
Sun, 26 Aug 2012 10:23:26 +0000 (22:23 +1200)
committerandy <andy@obsidian.(none)>
Sun, 26 Aug 2012 10:23:26 +0000 (22:23 +1200)
README
pd_readline2.c [new file with mode: 0644]
pdmenu5.c [new file with mode: 0644]
pdmenu6.c [new file with mode: 0644]
pdmenu7.c [new file with mode: 0644]

diff --git a/README b/README
index fa5a32296afd0d509f450c1c78fcde9664a998ac..9e5f3971d1468948618cd82bd6abf6aa3c44d995 100644 (file)
--- a/README
+++ b/README
@@ -12,6 +12,14 @@ This makes it VERY easy to see the various special flags
 in a keystroke sequence 
 (e.g. Esc, Ctrl, Alt and so on..... ).  
 
+Big breakthrough - the code can now distinguish between 
+normal "printable" keys and keys that should not be 
+printed (e.g. arrow keys, Ctrl sequences and so on).  
+
+The next goal is to make the code move the cursor to the 
+left (when left-arrow is pressed) and delete a char (when 
+backspace is pressed). 
+
 Still some way to go to get to a REAL readline, but we are 
 "on the way......"
 
diff --git a/pd_readline2.c b/pd_readline2.c
new file mode 100644 (file)
index 0000000..02b943d
--- /dev/null
@@ -0,0 +1,186 @@
+\r
+\r
+/*  pd_readline2.c                                          */  \r
+/*  Status (as at 26th Aug 2012) : useful progress.         */ \r
+/*  Keystroke sequences (along with the special flags       */ \r
+/*  like Esc, Ctrl, Alt etc are now stored in a buffer      */ \r
+/*  ( an array of structs ).                                */    \r
+\r
+/*  The code can now distinguish between a printable key    */ \r
+/*  and a key that should not be printed.                   */ \r
+\r
+/*  It will still be some time before this is a REAL        */ \r
+/*  readline, but we are "on the way"......                 */     \r
+/*  This code is released to the public domain.             */ \r
+/*  "Share and enjoy...."  ;)                               */  \r
+\r
+\r
+#include <string.h>   \r
+#include <stdio.h> \r
+#include <termios.h>  /* For getch()  */  \r
+\r
+/* This implementation of getch() is from here - */ \r
+/* http://wesley.vidiqatch.org/                  */ \r
+/* Thanks, Wesley!                               */  \r
+static struct termios old, new;\r
+\r
+/* Initialize new terminal i/o settings */\r
+void initTermios(int echo) {\r
+    tcgetattr(0, &old); /* grab old terminal i/o settings */\r
+    new = old; /* make new settings same as old settings */\r
+    new.c_lflag &= ~ICANON; /* disable buffered i/o */\r
+    new.c_lflag &= echo ? ECHO : ~ECHO; /* set echo mode */\r
+    tcsetattr(0, TCSANOW, &new); /* use these new terminal i/o settings now */\r
+}\r
+\r
+\r
+/* Restore old terminal i/o settings */\r
+void resetTermios(void) {\r
+    tcsetattr(0, TCSANOW, &old);\r
+}\r
+\r
+\r
+/* Read 1 character - echo defines echo mode */\r
+char getch_(int echo) {\r
+    char ch;\r
+    initTermios(echo);\r
+    ch = getchar();\r
+    resetTermios();\r
+    return ch;\r
+}\r
+\r
+\r
+/* Read 1 character without echo */\r
+char getch(void) {\r
+    return getch_(0);\r
+}\r
+\r
+\r
+/* Read 1 character with echo */\r
+char getche(void) {\r
+    return getch_(1);\r
+} \r
+\r
+\r
+\r
+/*  Helper function, to let us see if a .history file */ \r
+/*  exists in the current directory.  */ \r
+int fexists(char *fname)\r
+{  \r
+\r
+   FILE *fptr ;        \r
+       \r
+   fptr = fopen(fname, "r") ;  \r
+   \r
+   if ( !fptr )  return -1 ;  /* File does not exist in dir. */        \r
+        \r
+   fclose(fptr);  \r
+   return 0;    /* File DOES exist in dir.  */         \r
+\r
+} \r
+\r
+\r
+ /* Struct to store key sequences */ \r
+typedef struct { \r
+          int fnkey; \r
+          int ctrl; \r
+          int alt ; \r
+       int shf ; \r
+       int esc ; \r
+       int lbr ;  /* For left-bracket ([) of escape sequences */ \r
+       int key; \r
+   } keyseq ;         \r
+\r
+\r
+\r
+\r
+\r
+int main(void)\r
+{\r
+       \r
+  printf("Public Domain Readline \n");              \r
+      \r
+  /* Buffer - an array of keyseq structs.                     */  \r
+  /* Note - now that we store the keystrokes in here,         */ \r
+  /* we can look at the various flags and decide whether to   */ \r
+  /* "echo" the key (as normal) or suppress it (as with an    */ \r
+  /* arrow key).                                              */    \r
+  keyseq buffer[80] ;   \r
+  \r
+  /* Buffer "pointer"  */ \r
+  int bufpnt = 0; \r
+  \r
+  \r
+  /* Test for existence of history file. */  \r
+  int exists;  \r
+  exists = fexists(".history"); \r
+  printf("Result: %d \n", exists);  \r
+  \r
+  while(1) \r
+    {  \r
+                   \r
+      int key = getch(); \r
+       \r
+      /* Printable chars. */  \r
+      if ( (key >= 32)  && (key <= 126) ) \r
+      { \r
+               /* We have a printable key so print it. */   \r
+               putchar(key);                   \r
+        buffer[bufpnt].key = key;  \r
+        bufpnt += 1; \r
+      }  \r
+                                                                       \r
+      /* Up arrow is 27, 91, 65.    ( ESC [ A )   */   \r
+      /* Down arrow is 27, 91, 66.  ( ESC [ B )   */ \r
+      /* Right arrow is 27, 91, 67. ( ESC [ C )   */ \r
+      /* Left arrow is 27, 91, 68.  ( ESC [ D )   */    \r
+      /* Function keys.     */ \r
+      /* F2 is 27, 79, 81.  */  \r
+      /* F3 is 27, 79, 82.  */  \r
+      /* F4 is 27, 79, 83.  */  \r
+           \r
+                \r
+      else if(key == 27)        \r
+          {  \r
+                         \r
+                         buffer[bufpnt].esc = 1;                        \r
+                         key = getch(); \r
+              if(key == 91)               \r
+              buffer[bufpnt].lbr = 1;                \r
+              key = getch(); \r
+              if( (key >= 65) && (key <= 68) )  \r
+               {                                \r
+                                buffer[bufpnt].key = key;                  \r
+               }                                 \r
+            bufpnt += 1;           \r
+          } \r
+              \r
+                                          \r
+                                                 \r
+    /* The Enter key exits. Enter is 10 decimal */       \r
+        else if(key == 10)  \r
+               { \r
+                        puts("\n");  \r
+                        int j ; \r
+                  /* Print the array of structs. */ \r
+                    for (j=0; j<20; j++)       \r
+                    { \r
+                           printf("Fnkey: %d ", buffer[j].fnkey  ) ; \r
+                           printf("Ctrl:  %d ", buffer[j].ctrl   ) ;   \r
+                           printf("Alt:   %d ", buffer[j].alt    ) ; \r
+                           printf("Shf:   %d ", buffer[j].shf    ) ; \r
+                           printf("Esc:   %d ", buffer[j].esc    ) ; \r
+                           printf("Lbr:   %d ", buffer[j].lbr    ) ;  \r
+                   printf("Key:   %d \n", buffer[j].key  ) ; \r
+                }                      \r
+                                         \r
+             break;           \r
+         }  /* Key = Enter */   \r
+    }                \r
+                                      \r
+       return 0;\r
+}  \r
+\r
+\r
+\r
+\r
diff --git a/pdmenu5.c b/pdmenu5.c
new file mode 100644 (file)
index 0000000..9c8721f
--- /dev/null
+++ b/pdmenu5.c
@@ -0,0 +1,139 @@
+\r
+\r
+/*  pd_readline.c                                           */  \r
+/*  Status (as at 26th Aug 2012) : useful progress.         */ \r
+/*  Keystroke sequences (along with the special flags       */ \r
+/*  like Esc, Ctrl, Alt etc are now stored in a buffer      */ \r
+/*  ( an array of structs ).                                */    \r
+/*  It will still be some time before this is a REAL        */ \r
+/*  readline, but we are "on the way"......                 */     \r
+/*  This code is released to the public domain.             */ \r
+/*  "Share and enjoy...."  ;)                               */  \r
+\r
+\r
+#include <string.h>   \r
+#include <stdio.h> \r
+\r
+\r
+/*  Helper function, to let us see if a .history file */ \r
+/*  exists in the current directory.  */ \r
+int fexists(char *fname)\r
+{  \r
+\r
+   FILE *fptr ;        \r
+       \r
+   fptr = fopen(fname, "r") ;  \r
+   \r
+   if ( !fptr )  return -1 ;  /* File does not exist in dir. */        \r
+        \r
+   fclose(fptr);  \r
+   return 0;    /* File DOES exist in dir.  */         \r
+\r
+} \r
+\r
+\r
+ /* Struct to store key sequences */ \r
+typedef struct { \r
+          int fnkey; \r
+          int ctrl; \r
+          int alt ; \r
+       int shf ; \r
+       int esc ; \r
+       int lbr ;  /* For left-bracket ([) of escape sequences */ \r
+       int key; \r
+   } keyseq ;         \r
+\r
+\r
+\r
+\r
+\r
+int main(void)\r
+{\r
+       \r
+  printf("Public Domain Readline \n");              \r
+  printf("NOTE! - at the moment, we are using \n");  \r
+  printf("NON-echoing reads, storing the keystrokes \n");  \r
+  printf("in a buffer \n");   \r
+    \r
+  \r
+  /* Buffer - an array of keyseq structs.                     */  \r
+  /* Note - now that we store the keystrokes in here,         */ \r
+  /* we can look at the various flags and decide whether to   */ \r
+  /* "echo" the key (as normal) or suppress it (as with an    */ \r
+  /* arrow key).                                              */    \r
+  keyseq buffer[80] ;   \r
+  \r
+  /* Buffer "pointer"  */ \r
+  int bufpnt = 0; \r
+  \r
+  \r
+  /* Test for existence of history file. */  \r
+  int exists;  \r
+  exists = fexists(".history"); \r
+  printf("Result: %d \n", exists);  \r
+  \r
+  while(1) \r
+    {  \r
+                   \r
+      int key = getchar(); \r
+       \r
+      /* Printable chars. */  \r
+      if ( (key >= 32)  && (key <= 126) ) \r
+      { \r
+               putchar(key);                   \r
+        buffer[bufpnt].key = key;  \r
+        bufpnt += 1; \r
+      }  \r
+                                                                       \r
+      /* Up arrow is 27, 91, 65.    ( ESC [ A )   */   \r
+      /* Down arrow is 27, 91, 66.  ( ESC [ B )   */ \r
+      /* Right arrow is 27, 91, 67. ( ESC [ C )   */ \r
+      /* Left arrow is 27, 91, 68.  ( ESC [ D )   */    \r
+      /* Function keys.     */ \r
+      /* F2 is 27, 79, 81.  */  \r
+      /* F3 is 27, 79, 82.  */  \r
+      /* F4 is 27, 79, 83.  */  \r
+           \r
+                \r
+      else if(key == 27)        \r
+          {                      \r
+                         buffer[bufpnt].esc = 1;                        \r
+                         key = getchar(); \r
+              if(key == 91)               \r
+              buffer[bufpnt].lbr = 1;                \r
+              key = getchar(); \r
+              if( (key >= 65) && (key <= 68) )  \r
+               {                                \r
+                                buffer[bufpnt].key = key;                  \r
+               }                                 \r
+            bufpnt += 1;           \r
+          } \r
+              \r
+                                          \r
+                                                 \r
+    /* The Enter key exits. Enter is 10 decimal */       \r
+        else if(key == 10)  \r
+               { \r
+                        int j ; \r
+                  /* Print the array of structs. */ \r
+                    for (j=0; j<10; j++)       \r
+                    { \r
+                           printf("Fnkey: %d ", buffer[j].fnkey  ) ; \r
+                           printf("Ctrl:  %d ", buffer[j].ctrl   ) ;   \r
+                           printf("Alt:   %d ", buffer[j].alt    ) ; \r
+                           printf("Shf:   %d ", buffer[j].shf    ) ; \r
+                           printf("Esc:   %d ", buffer[j].esc    ) ; \r
+                           printf("Lbr:   %d ", buffer[j].lbr    ) ;  \r
+                   printf("Key:   %d \n", buffer[j].key  ) ; \r
+                }                      \r
+                                         \r
+             break;           \r
+         }  /* Key = Enter */   \r
+    }                \r
+                                      \r
+       return 0;\r
+}  \r
+\r
+\r
+\r
+\r
diff --git a/pdmenu6.c b/pdmenu6.c
new file mode 100644 (file)
index 0000000..08e6bc3
--- /dev/null
+++ b/pdmenu6.c
@@ -0,0 +1,154 @@
+\r
+\r
+/*  pd_readline.c                                           */  \r
+/*  Status (as at 26th Aug 2012) : useful progress.         */ \r
+/*  Keystroke sequences (along with the special flags       */ \r
+/*  like Esc, Ctrl, Alt etc are now stored in a buffer      */ \r
+/*  ( an array of structs ).                                */    \r
+/*  It will still be some time before this is a REAL        */ \r
+/*  readline, but we are "on the way"......                 */     \r
+/*  This code is released to the public domain.             */ \r
+/*  "Share and enjoy...."  ;)                               */  \r
+\r
+\r
+#include <string.h>   \r
+#include <stdio.h> \r
+\r
+\r
+/*  Helper function, to let us see if a .history file */ \r
+/*  exists in the current directory.  */ \r
+int fexists(char *fname)\r
+{  \r
+\r
+   FILE *fptr ;        \r
+       \r
+   fptr = fopen(fname, "r") ;  \r
+   \r
+   if ( !fptr )  return -1 ;  /* File does not exist in dir. */        \r
+        \r
+   fclose(fptr);  \r
+   return 0;    /* File DOES exist in dir.  */         \r
+\r
+} \r
+\r
+\r
+ /* Struct to store key sequences */ \r
+typedef struct { \r
+          int fnkey; \r
+          int ctrl; \r
+          int alt ; \r
+       int shf ; \r
+       int esc ; \r
+       int lbr ;  /* For left-bracket ([) of escape sequences */ \r
+       int key; \r
+   } keyseq ;         \r
+\r
+\r
+/* A custom getchar to test the character type. */ \r
+int mygetchar(void) \r
+{\r
+   int ch = getchar();          \r
+   if (  (ch >= 32)  && (ch <= 126) ) \r
+    { \r
+       putchar(ch); \r
+       return ch;   \r
+    } \r
+    else \r
+    { \r
+       return ch;          \r
+    }    \r
+   \r
+}\r
+\r
+\r
+\r
+int main(void)\r
+{\r
+       \r
+  printf("Public Domain Readline \n");              \r
+  printf("NOTE! - at the moment, we are using \n");  \r
+  printf("NON-echoing reads, storing the keystrokes \n");  \r
+  printf("in a buffer \n");   \r
+    \r
+  \r
+  /* Buffer - an array of keyseq structs.                     */  \r
+  /* Note - now that we store the keystrokes in here,         */ \r
+  /* we can look at the various flags and decide whether to   */ \r
+  /* "echo" the key (as normal) or suppress it (as with an    */ \r
+  /* arrow key).                                              */    \r
+  keyseq buffer[80] ;   \r
+  \r
+  /* Buffer "pointer"  */ \r
+  int bufpnt = 0; \r
+  \r
+  \r
+  /* Test for existence of history file. */  \r
+  int exists;  \r
+  exists = fexists(".history"); \r
+  printf("Result: %d \n", exists);  \r
+  \r
+  while(1) \r
+    {  \r
+                   \r
+      int key = mygetchar(); \r
+       \r
+      /* Printable chars. */  \r
+      if ( (key >= 32)  && (key <= 126) ) \r
+      { \r
+               putchar(key);                   \r
+        buffer[bufpnt].key = key;  \r
+        bufpnt += 1; \r
+      }  \r
+                                                                       \r
+      /* Up arrow is 27, 91, 65.    ( ESC [ A )   */   \r
+      /* Down arrow is 27, 91, 66.  ( ESC [ B )   */ \r
+      /* Right arrow is 27, 91, 67. ( ESC [ C )   */ \r
+      /* Left arrow is 27, 91, 68.  ( ESC [ D )   */    \r
+      /* Function keys.     */ \r
+      /* F2 is 27, 79, 81.  */  \r
+      /* F3 is 27, 79, 82.  */  \r
+      /* F4 is 27, 79, 83.  */  \r
+           \r
+                \r
+      else if(key == 27)        \r
+          {                      \r
+                         buffer[bufpnt].esc = 1;                        \r
+                         key = mygetchar(); \r
+              if(key == 91)               \r
+              buffer[bufpnt].lbr = 1;                \r
+              key = mygetchar(); \r
+              if( (key >= 65) && (key <= 68) )  \r
+               {                                \r
+                                buffer[bufpnt].key = key;                  \r
+               }                                 \r
+            bufpnt += 1;           \r
+          } \r
+              \r
+                                          \r
+                                                 \r
+    /* The Enter key exits. Enter is 10 decimal */       \r
+        else if(key == 10)  \r
+               { \r
+                        int j ; \r
+                  /* Print the array of structs. */ \r
+                    for (j=0; j<10; j++)       \r
+                    { \r
+                           printf("Fnkey: %d ", buffer[j].fnkey  ) ; \r
+                           printf("Ctrl:  %d ", buffer[j].ctrl   ) ;   \r
+                           printf("Alt:   %d ", buffer[j].alt    ) ; \r
+                           printf("Shf:   %d ", buffer[j].shf    ) ; \r
+                           printf("Esc:   %d ", buffer[j].esc    ) ; \r
+                           printf("Lbr:   %d ", buffer[j].lbr    ) ;  \r
+                   printf("Key:   %d \n", buffer[j].key  ) ; \r
+                }                      \r
+                                         \r
+             break;           \r
+         }  /* Key = Enter */   \r
+    }                \r
+                                      \r
+       return 0;\r
+}  \r
+\r
+\r
+\r
+\r
diff --git a/pdmenu7.c b/pdmenu7.c
new file mode 100644 (file)
index 0000000..02b943d
--- /dev/null
+++ b/pdmenu7.c
@@ -0,0 +1,186 @@
+\r
+\r
+/*  pd_readline2.c                                          */  \r
+/*  Status (as at 26th Aug 2012) : useful progress.         */ \r
+/*  Keystroke sequences (along with the special flags       */ \r
+/*  like Esc, Ctrl, Alt etc are now stored in a buffer      */ \r
+/*  ( an array of structs ).                                */    \r
+\r
+/*  The code can now distinguish between a printable key    */ \r
+/*  and a key that should not be printed.                   */ \r
+\r
+/*  It will still be some time before this is a REAL        */ \r
+/*  readline, but we are "on the way"......                 */     \r
+/*  This code is released to the public domain.             */ \r
+/*  "Share and enjoy...."  ;)                               */  \r
+\r
+\r
+#include <string.h>   \r
+#include <stdio.h> \r
+#include <termios.h>  /* For getch()  */  \r
+\r
+/* This implementation of getch() is from here - */ \r
+/* http://wesley.vidiqatch.org/                  */ \r
+/* Thanks, Wesley!                               */  \r
+static struct termios old, new;\r
+\r
+/* Initialize new terminal i/o settings */\r
+void initTermios(int echo) {\r
+    tcgetattr(0, &old); /* grab old terminal i/o settings */\r
+    new = old; /* make new settings same as old settings */\r
+    new.c_lflag &= ~ICANON; /* disable buffered i/o */\r
+    new.c_lflag &= echo ? ECHO : ~ECHO; /* set echo mode */\r
+    tcsetattr(0, TCSANOW, &new); /* use these new terminal i/o settings now */\r
+}\r
+\r
+\r
+/* Restore old terminal i/o settings */\r
+void resetTermios(void) {\r
+    tcsetattr(0, TCSANOW, &old);\r
+}\r
+\r
+\r
+/* Read 1 character - echo defines echo mode */\r
+char getch_(int echo) {\r
+    char ch;\r
+    initTermios(echo);\r
+    ch = getchar();\r
+    resetTermios();\r
+    return ch;\r
+}\r
+\r
+\r
+/* Read 1 character without echo */\r
+char getch(void) {\r
+    return getch_(0);\r
+}\r
+\r
+\r
+/* Read 1 character with echo */\r
+char getche(void) {\r
+    return getch_(1);\r
+} \r
+\r
+\r
+\r
+/*  Helper function, to let us see if a .history file */ \r
+/*  exists in the current directory.  */ \r
+int fexists(char *fname)\r
+{  \r
+\r
+   FILE *fptr ;        \r
+       \r
+   fptr = fopen(fname, "r") ;  \r
+   \r
+   if ( !fptr )  return -1 ;  /* File does not exist in dir. */        \r
+        \r
+   fclose(fptr);  \r
+   return 0;    /* File DOES exist in dir.  */         \r
+\r
+} \r
+\r
+\r
+ /* Struct to store key sequences */ \r
+typedef struct { \r
+          int fnkey; \r
+          int ctrl; \r
+          int alt ; \r
+       int shf ; \r
+       int esc ; \r
+       int lbr ;  /* For left-bracket ([) of escape sequences */ \r
+       int key; \r
+   } keyseq ;         \r
+\r
+\r
+\r
+\r
+\r
+int main(void)\r
+{\r
+       \r
+  printf("Public Domain Readline \n");              \r
+      \r
+  /* Buffer - an array of keyseq structs.                     */  \r
+  /* Note - now that we store the keystrokes in here,         */ \r
+  /* we can look at the various flags and decide whether to   */ \r
+  /* "echo" the key (as normal) or suppress it (as with an    */ \r
+  /* arrow key).                                              */    \r
+  keyseq buffer[80] ;   \r
+  \r
+  /* Buffer "pointer"  */ \r
+  int bufpnt = 0; \r
+  \r
+  \r
+  /* Test for existence of history file. */  \r
+  int exists;  \r
+  exists = fexists(".history"); \r
+  printf("Result: %d \n", exists);  \r
+  \r
+  while(1) \r
+    {  \r
+                   \r
+      int key = getch(); \r
+       \r
+      /* Printable chars. */  \r
+      if ( (key >= 32)  && (key <= 126) ) \r
+      { \r
+               /* We have a printable key so print it. */   \r
+               putchar(key);                   \r
+        buffer[bufpnt].key = key;  \r
+        bufpnt += 1; \r
+      }  \r
+                                                                       \r
+      /* Up arrow is 27, 91, 65.    ( ESC [ A )   */   \r
+      /* Down arrow is 27, 91, 66.  ( ESC [ B )   */ \r
+      /* Right arrow is 27, 91, 67. ( ESC [ C )   */ \r
+      /* Left arrow is 27, 91, 68.  ( ESC [ D )   */    \r
+      /* Function keys.     */ \r
+      /* F2 is 27, 79, 81.  */  \r
+      /* F3 is 27, 79, 82.  */  \r
+      /* F4 is 27, 79, 83.  */  \r
+           \r
+                \r
+      else if(key == 27)        \r
+          {  \r
+                         \r
+                         buffer[bufpnt].esc = 1;                        \r
+                         key = getch(); \r
+              if(key == 91)               \r
+              buffer[bufpnt].lbr = 1;                \r
+              key = getch(); \r
+              if( (key >= 65) && (key <= 68) )  \r
+               {                                \r
+                                buffer[bufpnt].key = key;                  \r
+               }                                 \r
+            bufpnt += 1;           \r
+          } \r
+              \r
+                                          \r
+                                                 \r
+    /* The Enter key exits. Enter is 10 decimal */       \r
+        else if(key == 10)  \r
+               { \r
+                        puts("\n");  \r
+                        int j ; \r
+                  /* Print the array of structs. */ \r
+                    for (j=0; j<20; j++)       \r
+                    { \r
+                           printf("Fnkey: %d ", buffer[j].fnkey  ) ; \r
+                           printf("Ctrl:  %d ", buffer[j].ctrl   ) ;   \r
+                           printf("Alt:   %d ", buffer[j].alt    ) ; \r
+                           printf("Shf:   %d ", buffer[j].shf    ) ; \r
+                           printf("Esc:   %d ", buffer[j].esc    ) ; \r
+                           printf("Lbr:   %d ", buffer[j].lbr    ) ;  \r
+                   printf("Key:   %d \n", buffer[j].key  ) ; \r
+                }                      \r
+                                         \r
+             break;           \r
+         }  /* Key = Enter */   \r
+    }                \r
+                                      \r
+       return 0;\r
+}  \r
+\r
+\r
+\r
+\r