AllBASIC Forum

BASIC User Group => Code Challenges => Topic started by: AIR on October 22, 2018, 12:07:41 PM

Title: Menu me this....
Post by: AIR on October 22, 2018, 12:07:41 PM
Okay, so here's a coding challenge that can be used in the real world.  I had to do this for multiple OS's for work.

It's a terminal-based front-end to back up user data (I'm excluding the backup part).

The idea is to present the user/technician with a menu were they can select a user to backup.

Requirements:


Here is an implementation in Bash (more to come):

Code: Bash
  1. #!/bin/bash
  2.  
  3. EXCLUDE='Shared|Guest'
  4.  
  5. function SelectFolder() {
  6.     # PARAMETERS:
  7.     #    $1 - Menu Title
  8.     #    $2 - User Folders Location
  9.  
  10.     clear
  11.     TITLE="$1"
  12.     TLEN=${#TITLE}
  13.  
  14.     echo "$TITLE"
  15.     printf %${TLEN}s | tr " " "="
  16.     echo
  17.  
  18.     USER_LOC=$(ls "$2" | egrep -v $EXCLUDE)
  19.  
  20.     PS3="Select User: "
  21.  
  22.     select USER_ACCOUNT in $USER_LOC; do
  23.         test -n "$USER_ACCOUNT" && break
  24.         echo ">>> Invalid Selection <<<"
  25.     done
  26. }
  27.  
  28. # change the second parameter depending on OS...
  29. SelectFolder "Armando's Test Menu" "/Users"
  30.  
  31. printf "\nYou Selected: $USER_ACCOUNT\n"
  32.  


AIR.
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 12:14:08 PM
PYTHON VERSION

(It's an OPEN FORUM after all!!!  8))

Code: Python
  1. #!/usr/bin/env python
  2.  
  3. import glob,os,sys,time
  4.  
  5. excluded = '''Shared
  6.              Guest'''.split()
  7.                          
  8. def clear():
  9.         sys.stderr.write("\x1b[2J\x1b[H")
  10.  
  11. def SelectFolder(menu_title,path):
  12.  
  13.         loop = True
  14.         selection = {}
  15.         x = [x for x in glob.glob(path+'/*') if os.path.isdir(x) and not (os.path.basename(x) in excluded)]
  16.         while loop:
  17.                 clear()
  18.                 print menu_title.center(30,"-")
  19.  
  20.                 for cnt,item in enumerate(x):
  21.                         print '{}) {}'.format(cnt+1, os.path.basename(item))
  22.                         selection[str(cnt+1)] = item
  23.  
  24.                 print "-"*30
  25.                 print
  26.                 choice = raw_input('Select a User Folder: ')
  27.  
  28.                 try:
  29.                         selected = selection[choice]
  30.                         loop = False
  31.                 except KeyError:
  32.                         print ">> Invalid Selection <<"
  33.                         time.sleep(1)
  34.                         continue
  35.  
  36.         return selected
  37.  
  38. # change second parameter according to OS...
  39. selected = SelectFolder(" Armando's Test Menu ",'/Users')
  40. print "\nYou Selected: %s" % os.path.basename(selected)
  41.  

AIR.
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 12:18:08 PM
And for all the BASIC enthusiasts, a BACON VERSION  ;D

Code: Text
  1. DECLARE Exclude$ = "Guest|Shared"
  2.  
  3. FUNCTION SelectFolder$(Title$, Folder$)
  4.     LOCAL Menu$ ASSOC STRING
  5.     LOCAL tLen,cnt,Amount TYPE int
  6.     LOCAL Border$ TYPE STRING
  7.     LOCAL Loop TYPE bool
  8.  
  9.     tLen = LEN(Title$)
  10.     Loop = TRUE
  11.     Border$ = FILL$(LEN(Title$), 45 )
  12.  
  13.     WHILE Loop
  14.         Count = 1
  15.         OPEN Folder$ FOR DIRECTORY AS mydir
  16.  
  17.         CLEAR
  18.  
  19.         PRINT Title$
  20.  
  21.         PRINT Border$
  22.  
  23.         REPEAT
  24.             GETFILE myfile$ FROM mydir
  25.             IF LEFT$(myfile$,1)== "." OR ISFALSE(LEN(myfile$)) OR REGEX(myfile$,Exclude$) THEN
  26.                 CONTINUE
  27.             ELSE
  28.                 Menu$(STR$(Count)) = myfile$
  29.                 PRINT Count,") ", myfile$
  30.                 INCR Count
  31.             ENDIF
  32.         UNTIL ISFALSE(LEN(myfile$))
  33.        
  34.         CLOSE DIRECTORY mydir
  35.  
  36.         PRINT Border$
  37.        
  38.         Amount = NRKEYS(Menu$)
  39.         PRINT
  40.  
  41.         INPUT "Select User: ",Selection$
  42.         blah = VAL(Selection$)
  43.         IF (blah < 1 OR blah > Amount) THEN
  44.             PRINT ">>> Invalid Selection <<"
  45.             SLEEP 1000
  46.             CONTINUE
  47.         ELSE
  48.             Loop = FALSE
  49.         ENDIF
  50.     WEND
  51.     RETURN Menu$(Selection$)
  52. END FUNCTION
  53.  
  54. selected$ = SelectFolder$("Armando's Test Menu","/Users")
  55.  
  56. PRINT
  57. PRINT "You Selected: ", selected$
  58.  

AIR.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 12:43:23 PM
Would it be okay if I submitted my Script BASIC cross platform command line backup select utility to accept a path/file mask as an argument so the utility could be used in a 'batch' file? STDIO
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 12:45:00 PM
Sure, as long as the core requirements are met.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 12:57:39 PM
I should have read your requirements better.  >:(

I thought you were looking for a frontend file select with a default prefix directory path.

I'll give your requirements a try.

Title: Re: Menu me this....
Post by: John on October 22, 2018, 02:20:18 PM
I gave your BaCon example a try but had to make the _Bool change and change /Users to /home.

Code: [Select]
    DECLARE Exclude$ = "Guest|Shared"

    FUNCTION SelectFolder$(Title$, Folder$)
        LOCAL Menu$ ASSOC STRING
        LOCAL tLen,cnt,Amount TYPE int
        LOCAL Border$ TYPE STRING
        LOCAL Loop TYPE _Bool

        tLen = LEN(Title$)
        Loop = TRUE
        Border$ = FILL$(LEN(Title$), 45 )

        WHILE Loop
            Count = 1
            OPEN Folder$ FOR DIRECTORY AS mydir

            CLEAR

            PRINT Title$

            PRINT Border$

            REPEAT
                GETFILE myfile$ FROM mydir
                IF LEFT$(myfile$,1)== "." OR ISFALSE(LEN(myfile$)) OR REGEX(myfile$,Exclude$) THEN
                    CONTINUE
                ELSE
                    Menu$(STR$(Count)) = myfile$
                    PRINT Count,") ", myfile$
                    INCR Count
                ENDIF
            UNTIL ISFALSE(LEN(myfile$))

            CLOSE DIRECTORY mydir

            PRINT Border$

            Amount = NRKEYS(Menu$)
            PRINT

            INPUT "Select User: ",Selection$
            blah = VAL(Selection$)
            IF (blah < 1 OR blah > Amount) THEN
                PRINT ">>> Invalid Selection <<"
                SLEEP 1000
                CONTINUE
            ELSE
                Loop = FALSE
            ENDIF
        WEND
        RETURN Menu$(Selection$)
    END FUNCTION

    selected$ = SelectFolder$("Armando's Test Menu","/home")

    PRINT
    PRINT "You Selected: ", selected$


jrs@jrs-laptop:~/BaCon/bacon-3.7.3$ ./air

Armando's Test Menu
-------------------
1) jrs
-------------------

Select User: 1

You Selected: jrs
jrs@jrs-laptop:~/BaCon/bacon-3.7.3$


This is a much easier task than I first imagined.
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 02:27:53 PM
I gave your BaCon example a try but had to make the _Bool change and change /Users to /home.

Hmm.  It worked with bool here on macOS and Raspbian, are you using the latest Bacon?

Quote from: John
This is a much easier task than I first imagined.

Yeah, it's all about putting the pieces together.  Once you get there, it's pretty easy when you look back...

AIR.
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 02:30:11 PM
NIM VERSION

Code: Text
  1. import os,ospaths,strutils,tables
  2.  
  3. proc input(prompt: string): int =
  4.     var
  5.         selection:string
  6.  
  7.     stdout.write prompt
  8.     selection = readLine(stdin)
  9.     return selection.parseInt
  10.  
  11.  
  12. proc SelectFolder(menu_title:string, path:string): string =
  13.     var
  14.         Count,Choice:int
  15.         Loop = true
  16.         userFolder,selected, Border:string
  17.         Exclude = ["Shared","Guest"]
  18.         menu = initTable[int, string]()
  19.    
  20.     Border = "-".repeat(menu_title.len)
  21.  
  22.     while Loop:
  23.         stdout.write "\x1b[2J\x1b[H"
  24.         echo menu_title
  25.         echo Border
  26.         Count = 1
  27.         for kind, path in walkDir(path):
  28.             userFolder = path.extractFilename
  29.             if userFolder.startsWith(".") or Exclude.contains(userFolder):
  30.                 continue
  31.             else:
  32.                 menu[Count] = userFolder
  33.                 echo Count,") ", userFolder
  34.                 Count += 1
  35.        
  36.         echo Border
  37.  
  38.         try:
  39.             Choice = input("Select a User Folder: ")
  40.             selected = menu[Choice]
  41.             Loop = false
  42.         except KeyError, ValueError:
  43.             echo ">> Invalid Selection <<"
  44.             sleep 1000
  45.             continue
  46.        
  47.     return selected
  48.  
  49. var selected = SelectFolder("Armando's Test Menu","/Users")
  50.  
  51. echo "\nSelected User: ",selected

AIR.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 02:42:24 PM
On Linux and I assume iOS one must run this as root.
Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 02:43:55 PM
On Linux and I assume iOS one must run this as root.

Not the menu.  If you go ahead and add code to do the actual backup, then yes.  But that's outside of the scope of this "challenge"...

BTW, root is not usually in the /home folder.  It's typically at /root in Linux, and in /var under macOS (not iOS).
Title: Re: Menu me this....
Post by: John on October 22, 2018, 04:12:52 PM
Quote from: AIR
BTW, root is not usually in the /home folder.  It's typically at /root in Linux, and in /var under macOS (not iOS).

Bad assumption on my part without checking first.

Here is the Script BASIC submission.

Code: ScriptBasic
  1. fn = FREEFILE()
  2. Exclude = "Guest|Shared"
  3. Title = "Armando's Test Menu"
  4. file_list = ""
  5. Border = STRING(LEN(Title), "-")
  6. Search_Options = SbCollectDirectories
  7. OPEN DIRECTORY "/home" PATTERN "" OPTION Search_Options AS #fn
  8. RESET DIRECTORY #fn
  9. fName = NEXTFILE(fn)
  10. WHILE fName <> undef
  11.   IF NOT INSTR(Exclude, fName) THEN file_list &= fName & "\n"
  12.   file_list &= fName & "\n"
  13.   fName = NEXTFILE(fn)
  14. WEND
  15. CLOSE DIRECTORY #fn
  16. SPLITA file_list BY "\n" TO users
  17. PRINT Title,"\n",Border,"\n"
  18. FOR idx = 0 TO UBOUND(users)
  19.   PRINT idx,") ",users[idx],"\n"
  20. NEXT
  21. PRINT Border,"\n","Select User: "
  22. LINE INPUT uid
  23. uid = VAL(CHOMP(uid))
  24. PRINT "You Selected: ", users[uid], "\n"
  25.  


jrs@jrs-laptop:~/sb/examples/test$ scriba air.sb
Armando's Test Menu
-------------------
0) jrs
-------------------
Select User: 0
You Selected: jrs
jrs@jrs-laptop:~/sb/examples/test$


Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 04:28:06 PM
Code: [Select]
Armando's Test Menu
-------------------
[0] .localized
[1] Shared
[2] Shared
[3] Guest
[4] Guest
[5] riveraa
-------------------
Select User: 9
You Selected: undef

It's not filtering plus I'm seeing dupes on my Mac.  Also missing the Warning message if the selection is out of range.  And requirement #2 isn't exactly met.\

Dupes are due to this:
Code: ScriptBasic
  1. WHILE fName <> undef
  2.   IF NOT INSTR(Exclude, fName) THEN file_list &= fName & "\n"
  3.   file_list &= fName & "\n"
  4.   fName = NEXTFILE(fn)
  5. WEND

You added to file_list twice.  LOL.  I hate when that happens!!

AIR.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 04:52:45 PM
It was going to be a IF / END IF but decided on one line and forgot to delete it.   ;D

Try this.

Code: ScriptBasic
  1. ' AIR - Select user for backup
  2.  
  3. FUNCTION SelectFolder(Title, Folder)
  4.   LOCAL fn, Exclude, Border, file_list, fName, users, idx, uid
  5.   fn = FREEFILE()
  6.   Exclude = "Guest|Shared"
  7.   Border = STRING(LEN(Title), "-")
  8.   file_list = ""
  9.   OPEN DIRECTORY Folder PATTERN "" OPTION SbCollectDirectories AS #fn
  10.   RESET DIRECTORY #fn
  11.   fName = NEXTFILE(fn)
  12.   WHILE fName <> undef
  13.     IF NOT INSTR(Exclude, fName) = undef THEN file_list &= fName & "\n"
  14.     fName = NEXTFILE(fn)
  15.   WEND
  16.   CLOSE DIRECTORY #fn
  17.   SPLITA file_list BY "\n" TO users
  18.   PRINT Title,"\n",Border,"\n"
  19.   FOR idx = 0 TO UBOUND(users)
  20.     PRINT idx,") ",users[idx],"\n"
  21.   NEXT
  22.   Enter:
  23.   PRINT Border,"\n","Select User: "
  24.   LINE INPUT uid
  25.   uid = VAL(CHOMP(uid))
  26.   IF uid > UBOUND(users) THEN
  27.     PRINT ">> Invalid Selection <<", "\n"
  28.     GOTO Enter
  29.   END IF
  30.   SelectFolder = users[uid]
  31. END FUNCTION
  32. selected = SelectFolder("Armando's Test Menu", "/home")
  33. PRINT "You Selected: ", selected, "\n"
  34.  


jrs@jrs-laptop:~/sb/examples/test$ scriba air.sb
Armando's Test Menu
-------------------
0) jrs
-------------------
Select User: 1
>> Invalid Selection <<
-------------------
Select User: 0
You Selected: jrs
jrs@jrs-laptop:~/sb/examples/test$ ls -l /home
total 12
drwxr-xr-x   2 root root 4096 Oct 22 16:48 Guest
drwxr-xr-x 133 jrs  jrs  4096 Oct 22 14:01 jrs
drwxr-xr-x   2 root root 4096 Oct 22 16:48 Shared
jrs@jrs-laptop:~/sb/examples/test$



Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 06:04:05 PM
Code: [Select]
Armando's Test Menu
-------------------
0) .localized
1) riveraa
-------------------
Select User:

That's what I get with your code on my Mac.

EDIT:  You also don't account for non-numeric input, so if I input a letter or a string of letters, the result is the first entry in the AssocArray.

I get the following with SB code that I wrote (I modified what you originally posted):

Code: [Select]
Armando's Test Menu
-------------------
1) riveraa
-------------------
Select User:

Here's my take on it, this version refreshes the screen after showing the warning message:

Code: ScriptBasic
  1. FUNCTION  SelectFolder(Title,Path)
  2.   fn = FREEFILE()
  3.   Exclude = "Guest|Shared|Public"
  4.   file_list = ""
  5.   myloop = 1
  6.   Border = STRING(LEN(Title), "-")
  7.   Search_Options = SbCollectDirectories
  8.  
  9.  
  10.   WHILE myloop
  11.     OPEN DIRECTORY Path PATTERN "" OPTION Search_Options AS #fn
  12.     RESET DIRECTORY #fn
  13.     fName = NEXTFILE(fn)
  14.  
  15.     WHILE fName <> undef
  16.       IF NOT INSTR(Exclude, fName) = undef THEN
  17.         IF LEFT(fName,1) <> "." THEN file_list &= fName & "\n"  
  18.       END IF
  19.       fName = NEXTFILE(fn)
  20.     WEND
  21.  
  22.     CLOSE DIRECTORY #fn
  23.  
  24.     SPLITA file_list BY "\n" TO users
  25.  
  26.     PRINT "\x1b[2J\x1b[H"
  27.  
  28.     PRINT Title,"\n",Border,"\n"
  29.  
  30.     FOR idx = 0 TO UBOUND(users)
  31.         PRINT idx+1,") ",users[idx],"\n"
  32.     NEXT
  33.  
  34.     PRINT Border,"\n","Select User: "
  35.  
  36.     LINE INPUT uid
  37.  
  38.     uid = VAL(CHOMP(uid-1))
  39.  
  40.     IF users[uid] = undef THEN
  41.       PRINT ">> Invalid Selection <<\n"
  42.       SLEEP 1
  43.       file_list = undef
  44.     ELSE
  45.       myloop = 0
  46.     END IF
  47.   WEND
  48.  
  49.   SelectFolder = users[uid]
  50. END FUNCTION
  51.  
  52. selected = SelectFolder("Armando's Test Menu", "/Users")
  53.  
  54. PRINT "\nYou Selected: ",selected,"\n"

AIR.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 07:14:36 PM
Here is my last version with your improvements running on Windows 7 32 bit OS. Something strange going on with returning lower case filenames The file explorer, DIR and SB.display different results. (another reason why I dislike Windows)

Code: ScriptBasic
  1. ' AIR - Select user for backup
  2.  
  3. FUNCTION SelectFolder(Title, Folder)
  4.   LOCAL fn, Exclude, Border, Search_Options, file_list, fName, users, idx, uid
  5.   fn = FREEFILE()
  6.   Exclude = "Default|Public"
  7.   Border = STRING(LEN(Title), "-")
  8.   file_list = ""
  9.   OPEN DIRECTORY Folder PATTERN "" OPTION SbCollectDirectories AS #fn
  10.   RESET DIRECTORY #fn
  11.   fName = NEXTFILE(fn)
  12.   WHILE fName <> undef
  13.     IF NOT INSTR(Exclude, fName) = undef THEN file_list &= fName & "\n"
  14.     fName = NEXTFILE(fn)
  15.   WEND
  16.   CLOSE DIRECTORY #fn
  17.   SPLITA file_list BY "\n" TO users
  18.   Retry:
  19.   PRINT Title,"\n",Border,"\n"
  20.   FOR idx = 0 TO UBOUND(users)
  21.     PRINT idx + 1,") ",users[idx],"\n"
  22.   NEXT
  23.   PRINT Border,"\n","Select User: "
  24.   LINE INPUT uid
  25.   uid = CHOMP(uid)
  26.   IF uid - 1 > UBOUND(users) OR users[uid - 1] = undef THEN
  27.     PRINT ">> Invalid Selection <<", "\n\n"
  28.     GOTO Retry
  29.   END IF
  30.   SelectFolder = users[uid -1]
  31. END FUNCTION
  32. PRINT "You Selected: ", SelectFolder("Armando's Test Menu", "/Users"), "\n"
  33.  

Title: Re: Menu me this....
Post by: AIR on October 22, 2018, 08:13:19 PM
Ruby Version

In case Tomaaz is lurking.... 8)

Code: Ruby
  1. #!/usr/bin/env ruby
  2.  
  3. def SelectFolder(title,path)
  4.     $menu = {}
  5.  
  6.     $Loop = 1
  7.     $Exclude = ['Shared','Guest','Public']
  8.     $Border = "-"*title.length
  9.  
  10.     while $Loop == 1
  11.         puts "\e[H\e[2J"
  12.         puts title
  13.         puts $Border
  14.         $Count = 1
  15.         for fPath in Dir.glob(path + '/*')
  16.             $fNAME = File.basename("#{fPath}")
  17.  
  18.             if not $Exclude.include? "#{$fNAME}"
  19.                 puts $Count.to_s + ") " + "#{$fNAME}"
  20.                 $menu[$Count] = "#{$fNAME}"
  21.                 $Count+=1
  22.             end
  23.         end
  24.         puts $Border
  25.         print "\nSelect User: "
  26.         choice = gets
  27.         $selected = $menu[choice.to_i]
  28.         if $selected
  29.             $Loop = 0
  30.         else
  31.             puts ">> Invalid Selection <<"
  32.             sleep 1
  33.             next
  34.         end
  35.  
  36.     end
  37.     return $selected
  38. end
  39.  
  40. $selected = SelectFolder("Armando's Test Menu","/Users")
  41.  
  42. puts "\nYou Selected: #{$selected}"

I really don't like the Ruby syntax, but to each his/her own....

AIR.
Title: Re: Menu me this....
Post by: John on October 22, 2018, 11:23:32 PM
I added an ANSI escape sequence to the Linux version to clear the screen if an invaild selection is made before retrying. It doesn't seem to work on Windows, surprised?

Code: ScriptBasic
  1. '' AIR - Select user for backup
  2.  
  3. FUNCTION SelectFolder(Title, Folder)
  4.   LOCAL fn, Exclude, Border, Search_Options, file_list, fName, users, idx, uid
  5.   fn = FREEFILE()
  6.   Exclude = "Guest|Shared"
  7.   Border = STRING(LEN(Title), "-")
  8.   file_list = ""
  9.   OPEN DIRECTORY Folder PATTERN "" OPTION SbCollectDirectories AS #fn
  10.   RESET DIRECTORY #fn
  11.   fName = NEXTFILE(fn)
  12.   WHILE fName <> undef
  13.     IF NOT INSTR(Exclude, fName) = undef THEN file_list &= fName & "\n"
  14.     fName = NEXTFILE(fn)
  15.   WEND
  16.   CLOSE DIRECTORY #fn
  17.   Retry:
  18.   SPLITA file_list BY "\n" TO users
  19.   PRINT Title,"\n",Border,"\n"
  20.   FOR idx = 0 TO UBOUND(users)
  21.     PRINT idx + 1,") ",users[idx],"\n"
  22.   NEXT
  23.   PRINT Border,"\n","Select User: "
  24.   LINE INPUT uid
  25.   uid = CHOMP(uid)
  26.   IF users[uid - 1] = undef THEN
  27.     PRINT "\033[H\033[2J"
  28.     UNDEF users
  29.     GOTO Retry
  30.   END IF
  31.   SelectFolder = users[uid -1]
  32. END FUNCTION
  33. Selected = SelectFolder("Armando's Test Menu", "/home")
  34. PRINT "You Selected: ", Selected, "\n"
  35.  

Title: Re: Menu me this....
Post by: AIR on October 23, 2018, 04:47:55 PM
I added an ANSI escape sequence to the Linux version to clear the screen if an invaild selection is made before retrying. It doesn't seem to work on Windows, surprised?

In Windows 10, there is supposed to be a registry flag you can set to enable the built-in support for ANSI sequences.  Or you can try loading up Ansi.sys (16 bit though!!) or one of the later Ansi.sys enhancements.

Or you can just exec the "cls" command...

John, does SB support conditional execution based on OS?  Like an #IFDEF in C?

AIR.
Title: Re: Menu me this....
Post by: John on October 23, 2018, 04:55:18 PM
Quote
John, does SB support conditional execution based on OS?  Like an #IFDEF in C?

The SB OPTION feature is a good example of conditional code execution. The SB ENVIRON() function returns a wealth of information about what you're running under. The conditional code execution could be based on that in BASIC.

I still have one more round for your challenge which is to give it an ANSI dress. (color / line graphics)

Any idea how you want to handle the Windows version of this and the issues I describe in a previous post?
Title: Re: Menu me this....
Post by: AIR on October 23, 2018, 06:09:58 PM
SB apparently doesn't honor the Hidden attribute for files/folders, so you'll have to add those "extra" directories to the filter list.

Title: Re: Menu me this....
Post by: John on October 23, 2018, 06:22:37 PM
Since SB Windows is returning file names in lowrer case, does it matter for the choice value returned?
Title: Re: Menu me this....
Post by: AIR on October 23, 2018, 07:04:13 PM
Not directly, since that is how SB is returning the directory name itself when in Windows. But it will affect the exclusions unless they are also all lowercase.

The conversion to lowercase is done in the file "filesys.c", in the "file_readdir" function (which I believe NEXTFILE actually calls).

Within that function, this line:

Code: C
  1. for( ; *s ; s++ )if( isupper(*s) )*s = tolower(*s);

Changes the returned filename to all lowercase.  This is only enabled when SB is compiled for Windows.

AIR.
Title: Re: Menu me this....
Post by: John on October 23, 2018, 07:12:49 PM
That's just the tip of the iceberg. What about unicode file names?
Title: Re: Menu me this....
Post by: AIR on October 23, 2018, 07:35:17 PM
If SB doesn't support unicode presently, you would have to alter at the very least all file-related functions so that they would use the "W" variant of the native C functions instead of the "A" (Ansi) variant which if I remember from my Windows days is the default. 

You would also probably need to alter some of the string functions and how the buffers themselves are allocated and removed.

Not a task for the faint of heart.  That's my ultra high level view, anyway.

But that conversation really doesn't belong in this thread.
Title: Re: Menu me this....
Post by: John on October 23, 2018, 07:58:24 PM
Oxygen Basic is currently going through that transistion.

This might be a way SB can deal with unicode.

https://www.gnu.org/software/libunistring/manual/
Title: Re: Menu me this....
Post by: AIR on October 24, 2018, 08:44:59 AM
FreePascal Version (Tested on macOS and Windows 10)

Code: Pascal
  1. Program MenuDemo;
  2. {$mode objfpc}{$H+}
  3.  
  4.  
  5. Uses SysUtils, StrUtils, Crt;
  6.  
  7. Function SelectFolder(Title,Path:String): String;
  8.     Var
  9.         Info : TSearchRec;
  10.         Count, Selection : Longint;
  11.         Menu  : array[0..14] of String;
  12.         Choice, Border: String;
  13.         Loop : Boolean = True;
  14.         Exclude: array[0..2] of String = ('Shared','Public','Guest');
  15.  
  16.     Begin
  17.         Repeat
  18.             ClrScr;
  19.             Border := DupeString('-',Length(Title));
  20.             Writeln(Title);
  21.             Writeln(Border);
  22.             Count:=0;
  23.             If FindFirst (Path + '/*',faDirectory,Info)=0 then begin
  24.                 Repeat
  25.                     With Info do begin
  26.                         If (Attr and faDirectory) = faDirectory then begin
  27.                             If ( LeftStr(Name,1) <> '.' ) AND ( AnsiIndexStr(Name,Exclude) = -1 ) then begin
  28.                                 Writeln (Count+1,') ',Name);
  29.                                 Menu[Count] := Name;
  30.                                 Inc(Count);
  31.                             end;
  32.                         end;
  33.                     end;
  34.                 Until FindNext(info) <> 0;
  35.             end;
  36.  
  37.             FindClose(Info);
  38.  
  39.             Writeln(Border);
  40.            
  41.             Write ('Select User: ');
  42.             ReadLn(Choice);
  43.  
  44.             try
  45.                 Selection := StrToInt(Choice);
  46.                 If (Selection < 1) Or (Selection > Count) Then
  47.                     raise Exception.Create('Out of Range');
  48.                 Loop := False;
  49.             except
  50.                 begin
  51.                     Writeln('>> Invalid Selection <<');
  52.                     Sleep (1000);
  53.                     Continue;
  54.                 end;
  55.             end;
  56.  
  57.         Until Loop = False;
  58.  
  59.         Result := menu[Selection-1];
  60. End;
  61.  
  62.  
  63. {** PROGRAM START **}
  64.  
  65. Var
  66.     Selected: String;
  67.  
  68. Begin
  69.     Selected := SelectFolder('Armando''s Test Menu', '/Users');
  70.  
  71.     Writeln(sLineBreak,'You Selected: ', Selected, sLineBreak);
  72. End.
  73.  


I haven't used Pascal in a LOOOOOONG time.  The last time was when I created an IDE for Hotbasic using Delphi...about 10 years ago.  It was fun gong back in time, so to speak...

AIR.
Title: Re: Menu me this....
Post by: John on October 24, 2018, 09:05:13 AM
I wish that wasted effort with Hot Basic could have been used to advance Script BASIC instead.  :-\
Title: Re: Menu me this....
Post by: AIR on October 24, 2018, 09:55:37 AM
I learned a LOT from Jim, both ASM _AND_ C as well as coding in general.  As much as he's always hated on C, he's very good at it.  Go figure.

I also learned a LOT from that community, which was always willing to answer questions and explain how they were able to enhance the language with their own crafted modules.

So not a wasted effort from my point of view, because if nothing else I was taught how NOT to wait for "someone else to do it".....

AIR.

Title: Re: Menu me this....
Post by: John on October 24, 2018, 10:07:38 AM
Experiance doesn't come cheap.
Title: Re: Menu me this....
Post by: AIR on October 24, 2018, 11:53:37 AM
HotBasic Version  8) 8) 8)

Code: Text
  1. $apptype console: $typecheck on
  2.  
  3. Declare function SelectFolder(Title$ as string, Path$ as string) as string
  4.  
  5. function SelectFolder(Title$ as string, Path$ as string) as string
  6.     defstr r$,border$
  7.     dim menu(15) as string*64
  8.     deflng count,selection,theLoop
  9.     dim exclude as LIST
  10.  
  11.     theLoop = 1
  12.     border$ = string$(Title$.length,"-")
  13.     exclude.AddItems("All Users","Default","Default User","Public")
  14.  
  15.     while theLoop
  16.         cls
  17.         count = 0
  18.  
  19.         print Title$
  20.         print border$
  21.  
  22.         r$ = dir$(Path$ + "/*",&H10)
  23.  
  24.         do
  25.             if r$[1] <> "." AND exclude.indexof(r$) = -1 then
  26.                 menu(count) = r$
  27.                 inc count
  28.                 print str$(count) + ") " + r$
  29.             end if
  30.             r$ = dir$
  31.         loop until r$.length = 0
  32.  
  33.         print border$
  34.         input "Select User: "; selection
  35.  
  36.         if selection < 1 or selection > count then
  37.             print ">> Invalid Selection <<"
  38.             sleep .8
  39.         else
  40.             theLoop = 0
  41.         end if
  42.     wend
  43.     result =  menu(selection-1)
  44. end function
  45.  
  46. ' ** PROGRAM START **
  47. defstr SelectedUser = SelectFolder("Armando's Test Menu","/Users")
  48. print CRLF + "You Selected: " + SelectedUser
  49.  


Man this takes me back.....

AIR.
Title: Re: Menu me this....
Post by: John on October 24, 2018, 02:53:00 PM
Wow!

It has been a long time since I seen Hot Basic code. Is Jim still in the game?

Curious, what are your thoughts about the unicode solution I proposed?
Title: Re: Menu me this....
Post by: AIR on October 24, 2018, 10:07:28 PM
C VERSION

Not 100% happy with this, but it seems to work okay...

Code: C
  1. #include <dirent.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <unistd.h>
  6.  
  7. #define NELEMS(x)  (sizeof(x) / sizeof((x)[0]))
  8.  
  9. const char* exclude[] = {"Guest","Shared","Public"};
  10.  
  11. char * SelectFolder(const char* Title, const char *Path){
  12.     char *line = NULL, *end, *menu[14];
  13.     char **ptr = menu;
  14.     int selected, count, i=0, loop = 1;
  15.     size_t len = 1, read = 0;
  16.  
  17.  
  18.     while (loop) {
  19.         printf("\033[2J\033[H");
  20.         printf("%s\n",Title);
  21.         printf("%.*s\n", (int)strlen(Title), "------------------------------------------");
  22.         i = count = 0;
  23.  
  24.         DIR *dir = opendir(Path);
  25.  
  26.         struct dirent *entry = readdir(dir);
  27.  
  28.         while (i< NELEMS(exclude)) {
  29.             while (entry != NULL) {
  30.                 if ( (entry->d_type == DT_DIR) && (strncmp(entry->d_name,".",1 ) != 0) && (strcmp(entry->d_name,exclude[i]) !=0)) {
  31.                     printf("%i) %s\n", count+1,entry->d_name);
  32.                     menu[count] = entry->d_name;
  33.                     count++;
  34.                 }
  35.                 entry = readdir(dir);
  36.                
  37.             }
  38.             i++;
  39.         }
  40.        
  41.         closedir(dir);
  42.  
  43.         printf("%.*s\n", (int)strlen(Title), "------------------------------------------");
  44.         printf("%s","Select User: ");
  45.         read = getline(&line, &len, stdin);
  46.         selected = strtol(line,&end,10);
  47.         printf("\n");
  48.        
  49.  
  50.         if ( (selected <1) || (selected > count) ) {
  51.             printf("%s\n",">> Invalid Selection <<");
  52.             sleep(1);
  53.         }else{
  54.             loop = 0;
  55.             free(line);
  56.         }
  57.            
  58.     }
  59.     return strdup(menu[selected-1]);
  60. }
  61.  
  62. int main() {
  63.     char* selected = SelectFolder("Armando's Test Menu","/Users");
  64.     printf("You Selected: %s\n\n",selected);
  65.     free(selected);
  66.     return 0;
  67. }
  68.  
  69.  

I think this is the last language I'll post...waiting on others to jump in....

AIR.
Title: Re: Menu me this....
Post by: John on October 24, 2018, 10:14:49 PM
Maybe Charles can find time to represent Oxygen Basic in this challenge?

Hopefully he will have better luck under Windows then I did with SB.
Title: Re: Menu me this....
Post by: AIR on October 25, 2018, 08:00:46 PM
Hopefully he will have better luck under Windows then I did with SB.

I don't see what the issue is with SB itself other than the forced lowercasing of the returned folder list.

IF you make the change in filesys.c that I suggested previously (just comment that line), and edit the script as I have below, it works fine:

Code: ScriptBasic
  1. ' AIR - Select user for backup
  2.    
  3. FUNCTION SelectFolder(Title, Folder)
  4.     LOCAL fn, Exclude, Border, Search_Options, file_list, fName, users, idx, uid, ret
  5.     fn = FREEFILE()
  6.  
  7.     Exclude = ".|All Users|Default|Default User|Public|desktop.ini"
  8.  
  9.     Border = STRING(LEN(Title), "-")
  10.  
  11.     OPEN DIRECTORY Folder PATTERN "" OPTION SbCollectDirectories AS #fn
  12.  
  13.     RESET DIRECTORY #fn
  14.  
  15.     fName = NEXTFILE(#fn)
  16.  
  17.     WHILE fName <> undef
  18.         IF NOT INSTR(Exclude, fName) = undef THEN file_list &= fName & "\n"
  19.         fName = NEXTFILE(#fn)
  20.     WEND
  21.  
  22.     CLOSE DIRECTORY #fn
  23.  
  24. Retry:
  25.     users = undef
  26.  
  27.     ret = EXECUTE("cmd /c cls",-1,PID)
  28.  
  29.     SPLITA file_list BY "\n" TO users
  30.    
  31.     PRINT Title,"\n",Border,"\n"
  32.  
  33.     FOR idx = 0 TO UBOUND(users)
  34.         PRINT idx + 1,") ",users[idx],"\n"
  35.     NEXT
  36.  
  37.     PRINT Border,"\n","Select User: "
  38.  
  39.     LINE INPUT uid
  40.  
  41.     uid = CHOMP(uid)
  42.  
  43.     IF uid - 1 > UBOUND(users) OR users[uid - 1] = undef THEN
  44.         PRINT "\n>> Invalid Selection <<"
  45.         SLEEP 1
  46.         GOTO Retry
  47.     END IF
  48.    
  49.     SelectFolder = users[uid -1]
  50.  
  51. END FUNCTION
  52.  
  53. selectedUser = SelectFolder("Armando's Test Menu", "/Users")
  54.  
  55. PRINT "\nYou Selected: ", selectedUser, "\n"
  56.    
  57.  

AIR.

P.S.  When posting code, please format it so that it's easier to read/follow (as I've done above).  It's easy to miss something, and also hard to follow the flow of the code, when it's densely packed.  Thanks!!

Title: Re: Menu me this....
Post by: John on October 25, 2018, 11:55:27 PM
Thanks for the confirmation your suggested change fixes the Windows issue. Why do you think Peter forced lower case for Windows only?

I will try to make the SB code I submit more readable. I may have caught Tomaaz's one line wonder disease.

Title: Re: Menu me this....
Post by: Mike Lobanovsky on October 26, 2018, 03:32:34 AM
... hard to follow the flow of the code, when it's densely packed.

Not so for me actually. I prefer camel case, 8-px monospace fonts, 2-char tabs, indentation guides and syntax highlighting, and I code on every line leaving one blank line between functions. :)
Title: Re: Menu me this....
Post by: Mike Lobanovsky on October 26, 2018, 03:47:48 AM
Why do you think Peter forced lower case for Windows only?

Obviously because Windows pathnames and filenames are case insensitive. Windows also supports forward and back slashes indiscriminately. It even allows them to be intermixed at will. :)
Title: Re: Menu me this....
Post by: John on October 26, 2018, 07:23:57 AM
Does that also hold true for unicode filenames?

I'm still going ahead with AIR's recommended change as it would be confusing to users seeing their filenames changed.
Title: Re: Menu me this....
Post by: Mike Lobanovsky on October 26, 2018, 12:21:31 PM
No, Unicode pathnames are a horror and you're done if you don't have an appropriate language support installed on your box. You won't be able to either access, or rename, or delete the offending file. But then, upper-/lower-casing isn't going to work for you either.

The same goes about Linux chars not allowed in the Windows pathnames created or copied to the Windows hosts from under their Linux VM's.

Avoid localized path/file names under Windows at all times and use ANSI whenever possible because there may be potential users whose environments are not equipped with the corresponding locale.

Not that it matters much to Anglo-Saxon users though whose locale support is built into the MS Windows installations natively. ;)
Title: Re: Menu me this....
Post by: John on October 26, 2018, 01:21:55 PM
Thanks Mike for your thoughts on unicode. I'm going take your advice as well.