QB64.org Forum

Active Forums => Programs => Topic started by: NOVARSEG on February 03, 2021, 09:25:35 pm

Title: BMP picture viewer
Post by: NOVARSEG on February 03, 2021, 09:25:35 pm
Basic code for a BMP picture viewer.  Works with 24 bit per pixel BMP.

Does not use _LOADIMAGE.   To use the program, enter the name of the BMP file you want to look at. 


Code: QB64: [Select]
  1.  
  2. INPUT " Enter name of BMP file "; i$
  3.  
  4. DIM tt AS STRING * 4
  5. DIM t AS STRING * 3
  6. DIM handle AS LONG
  7.  
  8. GET #1, 11, OF
  9. PRINT "Offset to picture data"; OF
  10.  
  11. GET #1, 15, L
  12. PRINT "Header size "; L
  13.  
  14. GET #1, , W
  15. PRINT "image width "; W
  16.  
  17. GET #1, , H
  18. PRINT "image height "; H
  19.  
  20. GET #1, , I
  21.  
  22.  
  23. GET #1, , I
  24. PRINT "bits per pixel "; I
  25.  
  26. IF I = 32 THEN PRINT "Can't process 32 bit per pixel BMP": END
  27.  
  28. handle = _NEWIMAGE(W, H, 32)
  29. a = _MEMIMAGE(handle)
  30.  
  31. FOR R = H - 1 TO 0 STEP -1
  32.  
  33.     FOR C = 1 TO W * 3 STEP 3
  34.         GET #1, R * W * 3 + C + OF, t
  35.         tt = t + CHR$(255)
  36.         _MEMPUT a, a.OFFSET + N, tt
  37.         N = N + 4
  38.     NEXT C
  39.  
  40.  
  41. SCREEN _NEWIMAGE(W, H, 32)
  42. _PUTIMAGE (0, 0), handle
  43.  
  44.  
  45.  







Title: Re: BMP picture viewer
Post by: SMcNeill on February 03, 2021, 09:36:04 pm
Will this work with compressed images?

And the only difference with 32-bit images is you just get 4 bytes instead of 3 and adding CHR$(255).
Title: Re: BMP picture viewer
Post by: SpriggsySpriggs on February 03, 2021, 09:37:27 pm
Neat. I'll take a look at this sometime
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 03, 2021, 09:51:51 pm
Right now code works with normal uncompressed images .   

Yep adding that 4th byte is important with 24 bit per pixel BMPs 


Title: Re: BMP picture viewer
Post by: NOVARSEG on February 03, 2021, 10:12:52 pm
Updated so code won't try to open other formats.

Code: QB64: [Select]
  1.  
  2. INPUT " Enter name of BMP file "; i$
  3.  
  4. DIM tt AS STRING * 4
  5. DIM t AS STRING * 3
  6. DIM handle AS LONG
  7.  
  8. GET #1, 11, OF
  9. PRINT "Offset to picture data"; OF
  10.  
  11. GET #1, 15, L
  12. PRINT "Header size "; L
  13.  
  14. GET #1, , W
  15. PRINT "image width "; W
  16.  
  17. GET #1, , H
  18. PRINT "image height "; H
  19.  
  20. GET #1, , I
  21.  
  22.  
  23. GET #1, , I
  24. PRINT "bits per pixel "; I
  25.  
  26. IF I = 32 THEN: _FULLSCREEN: PRINT "Can't process 32 bit per pixel BMP": CLOSE: END
  27. IF I <> 24 THEN: _FULLSCREEN: PRINT "Not a 24 bit per pixel BMP": CLOSE: END
  28.  
  29.  
  30.  
  31.  
  32. handle = _NEWIMAGE(W, H, 32)
  33. a = _MEMIMAGE(handle)
  34.  
  35. FOR R = H - 1 TO 0 STEP -1
  36.  
  37.     FOR C = 1 TO W * 3 STEP 3
  38.         GET #1, R * W * 3 + C + OF, t
  39.         tt = t + CHR$(255)
  40.         _MEMPUT a, a.OFFSET + N, tt
  41.         N = N + 4
  42.     NEXT C
  43.  
  44.  
  45. SCREEN _NEWIMAGE(W, H, 32)
  46. _PUTIMAGE (0, 0), handle
  47.  
  48.  
  49.  












Title: Re: BMP picture viewer
Post by: NOVARSEG on February 04, 2021, 02:38:54 am
Update

used the FILELIST function to display BMP files in a folder

just click with mouse to display BMP

Code: QB64: [Select]
  1. DIM filename AS STRING
  2. _TITLE "Demo of FILELIST$ Function by Ted Weissgerber 2010 (modified)"
  3. dst& = _NEWIMAGE(800, 600, 32) 'for console bitmap
  4. SCREEN dst&
  5.  
  6. PAINT (799, 599), _RGB(180, 180, 180)
  7. LINE (7, 15)-(648, 496), _RGB(0, 0, 80), BF
  8.  
  9. 'LOCATE 33, 10: PRINT "Do you want file display in a new window? (Y/N)"
  10. 'DO
  11. 'SLEEP
  12. 'yesno$ = UCASE$(INKEY$)
  13. 'LOOP UNTIL yesno$ <> ""
  14.  
  15. 'IF yesno$ = "Y" THEN x% = 0: y% = 0 ELSE x% = 8: y% = 16 'set parameters
  16.  
  17. x% = 8: y% = 16
  18. LOCATE 33, 10
  19. ' INPUT "Enter a File Type extension(*.BMP, *.TXT, etc): ", spec$
  20. 'spec$ = UCASE$(spec$)
  21. ''IF INSTR(spec$, "*.") = 0 THEN spec$ = "*." + spec$
  22. spec$ = "*.bmp"
  23. '<<<<<<< If x% and y% parameters are 0, the display is in separate window.>>>>>>
  24. filename = FILELIST$(spec$, x%, y%) '<<<<<<<< added display position parameters
  25.  
  26.  
  27. LOCATE 35, 10: PRINT filename$, LFN$
  28. 'SLEEP
  29.  
  30.  
  31.  
  32.  
  33. 'END
  34.  
  35. 'INPUT " Enter name of BMP file "; i$
  36. OPEN filename FOR BINARY AS #1
  37.  
  38. DIM tt AS STRING * 4
  39. DIM t AS STRING * 3
  40. DIM handle AS LONG
  41.  
  42. GET #1, 11, OF
  43. PRINT "Offset to picture data"; OF
  44.  
  45. GET #1, 15, L
  46. PRINT "Header size "; L
  47.  
  48. GET #1, , W
  49. PRINT "image width "; W
  50.  
  51. GET #1, , H
  52. PRINT "image height "; H
  53.  
  54. GET #1, , I
  55.  
  56.  
  57. GET #1, , I
  58. PRINT "bits per pixel "; I
  59.  
  60. IF I = 32 THEN: _FULLSCREEN: PRINT "Can't process 32 bit per pixel BMP": CLOSE: END
  61. IF I <> 24 THEN: _FULLSCREEN: PRINT "Not a 24 bit per pixel BMP": CLOSE: END
  62.  
  63.  
  64.  
  65.  
  66. handle = _NEWIMAGE(W, H, 32)
  67. a = _MEMIMAGE(handle)
  68.  
  69. FOR R = H - 1 TO 0 STEP -1
  70.  
  71.     FOR C = 1 TO W * 3 STEP 3
  72.         GET #1, R * W * 3 + C + OF, t
  73.         tt = t + CHR$(255)
  74.         _MEMPUT a, a.OFFSET + N, tt
  75.         N = N + 4
  76.     NEXT C
  77.  
  78.  
  79. SCREEN _NEWIMAGE(W, H, 32)
  80. _PUTIMAGE (0, 0), handle
  81.  
  82.  
  83.  
  84. FUNCTION FILELIST$ (Spec$, xpos%, ypos%)
  85.     SHARED Path$, LFN$ 'values also accessable by program
  86.     REDIM LGFN$(25), SHFN$(25), Last$(25), DIR$(25), Paths$(25) '<<<< $DYNAMIC only
  87.     IF LEN(ENVIRON$("OS")) = 0 THEN EXIT FUNCTION 'DIR X cannot be used on Win 9X
  88.     f% = FREEFILE
  89.     IF xpos% + ypos% = 0 THEN '****************** Root path TITLE in separate window only
  90.         SHELL _HIDE "CD > D0S-DATA.INF"
  91.         OPEN "D0S-DATA.INF" FOR INPUT AS #f%
  92.         LINE INPUT #f%, current$
  93.         CLOSE #f%
  94.     END IF ' ******************************************** END TITLE(see _TITLE below)
  95.     Spec$ = UCASE$(LTRIM$(RTRIM$(Spec$)))
  96.     IF INSTR(Spec$, "/A:D") OR INSTR(Spec$, "/O:G") THEN
  97.         DL$ = "DIR": BS$ = "\" 'directory searches only
  98.     ELSE: DL$ = SPACE$(3): BS$ = ""
  99.     END IF
  100.     mode& = _COPYIMAGE(0) 'save previous screen value to restore if files displayed.
  101.     ' Get Specific file information if available
  102.     SHELL _HIDE "cmd /c dir " + Spec$ + " /X > D0S-DATA.INF" 'get data
  103.     Head$ = "      Short Name          Long Name                     Last Modified     "
  104.     tmp$ = " \ \  \          \   \                              \ \                  \"
  105.     OPEN "D0S-DATA.INF" FOR INPUT AS #f% 'read the data file
  106.     DO UNTIL EOF(f%)
  107.         LINE INPUT #f%, line$
  108.         IF INSTR(line$, ":\") THEN
  109.             Path$ = MID$(line$, INSTR(line$, ":\") - 1)
  110.             IF RIGHT$(Path$, 1) <> "\" THEN Path$ = Path$ + "\"
  111.             setcode% = 0: filecode% = 0
  112.         END IF
  113.         IF LEN(line$) > 25 AND MID$(line$, 1, 1) <> " " THEN 'don't read other info
  114.             IF format% = 0 THEN
  115.                 IF MID$(line$, 20, 1) = "M" OR INSTR(line$, "<") = 25 THEN
  116.                     Sst% = 40: Lst% = 53: Dst% = 26: format% = 1 ' XP
  117.                 ELSE: Sst% = 37: Lst% = 50: Dst% = 23: format% = 2 'VISTA
  118.                 END IF
  119.             END IF
  120.             IF LEN(line$) >= Lst% THEN filecode% = ASC(UCASE$(MID$(line$, Lst%, 1))) ELSE filecode% = 0
  121.             D1R$ = MID$(line$, Dst%, 3) 'returns directories only with Spec$ = "/A:D" or "/O:G"
  122.             IF D1R$ <> "DIR" THEN D1R$ = SPACE$(3) 'change if anything else
  123.             IF D1R$ = DL$ AND filecode% >= setcode% THEN
  124.                 cnt% = cnt% + 1
  125.                 DIR$(cnt%) = D1R$: Paths$(cnt%) = Path$
  126.                 Last$(cnt%) = MID$(line$, 1, 20)
  127.                 IF MID$(line$, Sst%, 1) <> SPACE$(1) THEN
  128.                     SHFN$(cnt%) = MID$(line$, Sst%, INSTR(Sst%, line$, " ") - Sst%)
  129.                     LGFN$(cnt%) = MID$(line$, Lst%)
  130.                 ELSE: SHFN$(cnt%) = MID$(line$, Lst%): LGFN$(cnt%) = ""
  131.                 END IF
  132.                 IF LEN(Spec$) AND (Spec$ = UCASE$(SHFN$(cnt%)) OR Spec$ = UCASE$(LGFN$(cnt%))) THEN
  133.                     Spec$ = SHFN$(cnt%) + BS$: FILELIST$ = Spec$: LFN$ = LGFN$(cnt%) + BS$
  134.                     noshow = -1: GOSUB KILLdata ' verifies file exist query (no display)
  135.                 END IF
  136.                 IF page% > 0 THEN ' pages after first
  137.                     IF cnt% = 1 THEN GOSUB NewScreen
  138.                     COLOR 11: LOCATE , 3: PRINT USING tmp$; DIR$(cnt%); SHFN$(cnt%); LGFN$(cnt%); Last$(cnt%)
  139.                     IF DIR$(cnt%) = "DIR" AND LEFT$(SHFN$(cnt%), 1) = "." THEN SHFN$(cnt%) = "": LGFN$(cnt%) = ""
  140.                 ELSE 'first page = 0
  141.                     IF cnt% = 2 THEN 'only display to screen if 2 or more files are found
  142.                         FList& = _NEWIMAGE(640, 480, 256)
  143.                         IF xpos% + ypos% > 0 THEN ' user wants display on program screen
  144.                             Show& = _NEWIMAGE(640, 480, 256) '<<<<<< ONSCREEN program displays only
  145.                             _DEST FList&
  146.                         ELSE: SCREEN FList& ' <<<<<<<<< Separate Window
  147.                             _TITLE current$ '<<<<<<<<<<<<<<<<<<<<<<<<<<< TITLE optional
  148.                         END IF
  149.                         GOSUB NewScreen '<<<< update function's screen with Putimage(see notes)
  150.                         COLOR 11: LOCATE , 3: PRINT USING tmp$; DIR$(1); SHFN$(1); LGFN$(1); Last$(1)
  151.                         IF DIR$(1) = "DIR" AND LEFT$(SHFN$(1), 1) = "." THEN SHFN$(1) = "": LGFN$(1) = ""
  152.                     END IF
  153.                     IF cnt% > 1 THEN
  154.                         COLOR 11: LOCATE , 3: PRINT USING tmp$; DIR$(cnt%); SHFN$(cnt%); LGFN$(cnt%); Last$(cnt%)
  155.                         IF DIR$(cnt%) = "DIR" AND LEFT$(SHFN$(cnt%), 1) = "." THEN SHFN$(cnt%) = "": LGFN$(cnt%) = ""
  156.                     END IF
  157.                 END IF 'page%
  158.                 IF cnt% MOD 25 = 0 THEN 'each page holds 25 file names
  159.                     COLOR 14: LOCATE 28, 24: PRINT "Select file or click here for next.";
  160.                     GOSUB pickfile
  161.                     page% = page% + 1: cnt% = 0
  162.                     REDIM LGFN$(25), SHFN$(25), Last$(25), DIR$(25), Paths$(25) '<<<< $DYNAMIC only
  163.                 END IF 'mod  25
  164.             END IF 'DIR = DL$
  165.         END IF 'len line$ > 25
  166.     LOOP
  167.     CLOSE #f%
  168.     last = 1: total% = cnt% + (page% * 25)
  169.     IF total% = 0 THEN FILELIST$ = "": Spec$ = "": LFN$ = "": noshow = -1: GOSUB KILLdata: 'no files(no display)
  170.     IF total% = 1 THEN 'one file(no display)
  171.         Spec$ = SHFN$(1) + BS$: FILELIST$ = Spec$: LFN$ = LGFN$(1) + BS$: noshow = -1: GOSUB KILLdata
  172.     END IF
  173.     IF DL$ = SPACE$(3) THEN
  174.         COLOR 10: LOCATE 28, 65: PRINT total%; "Files"
  175.     ELSE: COLOR 10: LOCATE 28, 65: PRINT total%; "Folders"
  176.     END IF
  177.     COLOR 14: LOCATE 28, 24: PRINT "Select file or click here to Exit. ";
  178.     pickfile:
  179.     _DEST FList&
  180.     ShowPath$ = RIGHT$(Path$, 78)
  181.     COLOR 15: LOCATE 29, 41 - (LEN(ShowPath$) \ 2): PRINT ShowPath$;
  182.     GOSUB NewDisplay
  183.     DO: Key$ = UCASE$(INKEY$): _LIMIT 30
  184.         DO WHILE _MOUSEINPUT
  185.             Tcol% = ((_MOUSEX - xpos%) \ 8) + 1 'mouse column with Putimage offset
  186.             Trow% = ((_MOUSEY - ypos%) \ 16) + 1 'mouse row with offset
  187.             Pick = _MOUSEBUTTON(1) ' get left button selection click
  188.         LOOP
  189.         IF Trow% > 2 AND Trow% < cnt% + 3 AND Tcol% > 0 AND Tcol% < 80 THEN 'when mouse in area
  190.             R% = Trow% - 2
  191.             IF P% = 0 OR P% > cnt% + 3 THEN P% = R%
  192.             IF P% = R% THEN
  193.                 COLOR 15: LOCATE R% + 2, 3: PRINT USING tmp$; DIR$(R%); SHFN$(R%); LGFN$(R%); Last$(R%)
  194.             ELSE
  195.                 COLOR 11: LOCATE P% + 2, 3: PRINT USING tmp$; DIR$(P%); SHFN$(P%); LGFN$(P%); Last$(P%)
  196.             END IF
  197.             GOSUB NewDisplay
  198.             P% = R%
  199.             IF Pick THEN
  200.                 Spec$ = SHFN$(R%)
  201.                 IF LEN(Spec$) THEN
  202.                     COLOR 13: LOCATE R% + 2, 3: PRINT USING tmp$; DIR$(R%); SHFN$(R%); LGFN$(R%); Last$(R%)
  203.                     GOSUB NewDisplay
  204.                     Spec$ = Spec$ + BS$: FILELIST$ = Spec$: Path$ = Paths$(R%)
  205.                     IF LEN(LGFN$(R%)) THEN LFN$ = LGFN$(R%) + BS$ ELSE LFN$ = ""
  206.                     _DELAY 1.5: CLS: SCREEN mode&: GOSUB KILLdata 'exit if user selection
  207.                 END IF
  208.             END IF 'len spec
  209.         END IF 'pick
  210.         IF LEN(Key$) THEN usercode% = ASC(Key$) ELSE usercode% = 0
  211.         IF usercode% > setcode% THEN setcode% = usercode% 'user can press letter to jump to
  212.         IF Pick AND Trow% > 27 THEN EXIT DO
  213.     LOOP UNTIL LEN(Key$)
  214.     _DELAY .4 'adjust delay for page scroll speed
  215.     DO: Key$ = INKEY$: LOOP UNTIL Key$ = ""
  216.     IF last = 0 THEN RETURN 'exit if file no more data
  217.     FILELIST$ = "": Spec$ = "": LFN$ = ""
  218.     CLS: SCREEN mode& 'resets program screen to previous condition
  219.     KILLdata:
  220.     CLOSE #f%: KILL "D0S-DATA.INF" 'kill D0S-DATA.INF file and exit
  221.     IF FList& < -1 THEN _FREEIMAGE FList&
  222.     IF Show& < -1 THEN _FREEIMAGE Show&
  223.     IF noshow = -1 AND mode& < -1 THEN _FREEIMAGE mode&
  224.     _AUTODISPLAY 'reset default settings
  225.     RETURN
  226.     NewScreen: 'clear screen and set display format
  227.     LINE (0, 0)-(639, 499), 0, BF
  228.     COLOR 14: LOCATE 2, 3: PRINT Head$
  229.     LINE (4, 4)-(636, 476), 13, B: LINE (5, 5)-(635, 475), 13, B
  230.     GOSUB NewDisplay
  231.     RETURN
  232.     NewDisplay: 'show program or window displays
  233.     IF xpos% + ypos% > 0 THEN
  234.         _PUTIMAGE , FList&, Show&
  235.         _DEST 0: _PUTIMAGE (xpos%, ypos%), Show&: _DISPLAY
  236.         _DEST FList&
  237.     ELSE: _DISPLAY
  238.     END IF
  239.     RETURN
  240.  






Title: Re: BMP picture viewer
Post by: bplus on February 04, 2021, 02:06:59 pm
@NOVARSEG

OK you fixed the first error on title but I am still getting path not found line 39

Code: QB64: [Select]
  1. spec$ = "*.*"
  2. '<<<<<<< If x% and y% parameters are 0, the display is in separate window.>>>>>>
  3. filename = FILELIST$(spec$, x%, y%) '<<<<<<<< added display position parameters
  4.  

so filename is missing, so maybe it's the spec$, changed to "*.*" takes care of that error (when no .bmp file is found) and what a lovely display Ted W! from 2010 but then I want to try another directory for bmp files I click where it says to change directory error again. I click current folder to change that and another error.
Not fun... :(

There is a tool to get Folders and Files that works cross platform from which you could work out file select or Folder change Best Answer here: https://www.qb64.org/forum/index.php?topic=1511.msg107143#msg107143

You would build your own filter for BMP files and the code for displaying the Directory Lists and File Lists, it's good exercise and you explain your code better to others and fix errors. There are also tools for displaying lists and selecting also probably harder to do than getting Folder and File lists.

Just trying to help, I don't save BMP's in this format but you are attempting a very common and useful task browsing for filename to load. It is important.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 04, 2021, 06:37:15 pm
@bplus
The code is buggy

When I paste a single bmp into a folder  or no BMP in folder it crashes

When  I paste 2 bmp into a folder it works.

Title: Re: BMP picture viewer
Post by: bplus on February 04, 2021, 07:28:53 pm
@NOVARSEG  buggy program is OK, just a little warning first please. Sorry I thought you were showing a demo.

I still recommend using someone else filename retriever, not Ted W from 2010. Someone here and now at forum that can help with code they wrote if you get stuck while using.

Steve has good filename retriever, Spriggsy, Dav... not just me to choose more current code from.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 04, 2021, 07:40:31 pm
The code is a demo used with BMP viewer. 

 Followed that link  https://www.qb64.org/forum/index.php?topic=1511.msg107143#msg107143 and found the code with library functions.   I have also done the same thing with the DOS interrupts.
Title: Re: BMP picture viewer
Post by: bplus on February 04, 2021, 07:49:23 pm
The code is a demo used with BMP viewer. 

 Followed that link  https://www.qb64.org/forum/index.php?topic=1511.msg107143#msg107143 and found the code with library functions.   I have also done the same thing with the DOS interrupts.

The advantage of using code from that link is that it is cross platform not Windows dependent so build a filename retriever from that and it can be used in Linux and Mac (so I hear). And reused for any app that needs to get a filename.

But maybe those other OS don't use that form of BMP. :)
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 04, 2021, 09:45:47 pm
The BMP format is cross platform. It's disadvantage is size of file.

Im trying to work with win API.  I'm trying to download  fileapi.h  not much luck so far.

https://docs.microsoft.com/en-us/windows/win32/api/fileapi/

Are you actually clicking the file name with mouse?

Microsoft recommends C++ when coding with WINAPI 
https://docs.microsoft.com/en-us/windows/win32/learnwin32/learn-to-program-for-windows

Can we do the same with QB64?
Title: Re: BMP picture viewer
Post by: bplus on February 04, 2021, 10:24:16 pm
The BMP format is cross platform. It's disadvantage is size of file.

Im trying to work with win API.  I'm trying to download  fileapi.h  not much luck so far.

https://docs.microsoft.com/en-us/windows/win32/api/fileapi/

Are you actually clicking the file name with mouse?

Microsoft recommends C++ when coding with WINAPI 
https://docs.microsoft.com/en-us/windows/win32/learnwin32/learn-to-program-for-windows

Can we do the same with QB64?

Windows API isn't going to help you in Linux.

Here is Tiny Navigator for File Retrieval, you need direntry.h in the same folder as QB64.exe.

Tiny Navigator for File Retrieval the left side is directories, the right is a sample of files from the directory highlighted. When on the directory you want to select a file from press f then another scrollable window comes up on right for file selection.

You can scroll, arrow, pageup and down by clicking green bars home end for start of array and end, you can also select directories and files by typing the number then enter.
Code: QB64: [Select]
  1. ' B+ started 2019-08-22  restart 2020-06-24
  2. ' 2019-08-22 orig post at https://www.qb64.org/forum/index.php?topic=1646.msg108682#msg108682
  3. ' 2019-08-23_13-25 try fix (to nav all directories) with one place for tmpFile, theory can't write files in some dir's
  4. ' For some reason Windows won't write to a fully pathed file in my user folder???from a SHELL command
  5. ' Try testing if dir exists "C:\temp" exists and making one if not, yes! and I can write my temp files there
  6. ' and now I can chDir anywhere!!!!
  7. ' 2019-08-23_14-25 Take Steve's Advice to use users tmp directory for temp files
  8. ' 2019-08-23_23+ Have files window working for file selection
  9.  
  10. ' 2020-06-24 restart using DirEntry.h, wow looks like Steve's fix helped ALLOT! Thanks SMcNeill
  11. ' I am hitting glitches where Windows wont let me access I think. So I am offering a c enter option
  12. ' to restart at the root but this might not work cross platform.
  13.  
  14. ' 2020-06-26 another fix to DirEntry.h usage add CHR$(0) to string when calling a C program. This fixed a snag
  15. ' between marka and next folder up when heading to root with .. calls.
  16.  
  17. ' 2020-07-04 Use this to Kill some pesky .ogg files
  18.  
  19. ' direntry.h needs to be in QB64 folder ' <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< IMPORTANT
  20.     FUNCTION load_dir& (s AS STRING)
  21.     FUNCTION has_next_entry& ()
  22.     SUB close_dir ()
  23.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  24.  
  25.  
  26. _TITLE "Tiny Navigator File Name Retriever:  press f to start File Selection List of current directory"
  27. SCREEN _NEWIMAGE(1000, 600, 32)
  28. _SCREENMOVE 100, 50
  29.  
  30. ' for GetFileName$()
  31. DIM SHARED selectedFile AS STRING
  32. REDIM SHARED DIRs(0) AS STRING, FILs(0) AS STRING
  33.  
  34. DIM done, myFile$, again$
  35. COLOR &HFF000000, &HFFFFFFFF
  36. DO 'test our new function
  37.     scnState 0
  38.     myFile$ = GetFileName$
  39.     scnState -1
  40.     CLS
  41.     PRINT "You selected: "; myFile$
  42.     PRINT
  43.     INPUT "Test another File Selection (y for yes) "; again$
  44.     IF again$ <> "y" THEN done = -1
  45. LOOP UNTIL done
  46.  
  47. FUNCTION GetFileName$ ()
  48.  
  49.     'you might try srnState before and after calling this function to restore conditions
  50.     ' uses: SUB GetLists (SearchDirectory AS STRING, DirList() AS STRING, FileList() AS STRING)  for getting dirs a nd files
  51.     ' uses: FUNCTION getArrayItemNumber& (locateRow, locateColumn, boxWidth, boxHeight, arr() AS STRING) for display large arrays and selecting
  52.  
  53.     ' share these with Main Code section:
  54.     ' 'for GetFileName$() and GetLists
  55.     'DIM SHARED selectedFile AS STRING
  56.     'REDIM SHARED DIRs(0) AS STRING, FILs(0) AS STRING
  57.  
  58.     DIM mySelection&, done$, i, t$
  59.  
  60.     _TITLE "Tiny Navigator File Name Retriever:  press f to start File Selection List of current directory"
  61.     DO
  62.         COLOR _RGB32(180, 180, 255), &HFF000000
  63.         CLS
  64.         t$ = "Current Directory: " + _CWD$
  65.         LOCATE 2, (_WIDTH / 8 - LEN(t$)) / 2: PRINT t$
  66.         REDIM DIRs(0) AS STRING, FILs(0) AS STRING
  67.         GetLists _CWD$, DIRs(), FILs()
  68.         FOR i = 0 TO UBOUND(FILs) ' this just offers a sample listing of files
  69.             IF i < 30 THEN LOCATE i + 4, 60: PRINT FILs(i) ELSE EXIT FOR
  70.         NEXT
  71.         mySelection& = getArrayItemNumber&(5, 5, 50, 30, DIRs())
  72.  
  73.         CLS
  74.         IF selectedFile <> "" THEN
  75.             GetFileName$ = selectedFile: done$ = "y"
  76.         ELSEIF mySelection& <> -1719 THEN
  77.             IF _TRIM$(DIRs(mySelection&)) <> "" THEN
  78.                 CHDIR DIRs(mySelection&)
  79.             END IF
  80.         ELSE
  81.             GetFileName$ = "": done$ = "y" ' cancelled or error
  82.         END IF
  83.         selectedFile = ""
  84.         _LIMIT 60
  85.     LOOP UNTIL done$ <> ""
  86.  
  87. SUB GetLists (SearchDirectory AS STRING, DirList() AS STRING, FileList() AS STRING)
  88.  
  89.     'put this block in main code section of your program close to top
  90.     '' direntry.h needs to be in QB64 folder
  91.     'DECLARE CUSTOMTYPE LIBRARY ".\direntry"
  92.     '    FUNCTION load_dir& (s AS STRING)
  93.     '    FUNCTION has_next_entry& ()
  94.     '    SUB close_dir ()
  95.     '    SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  96.     'END DECLARE
  97.  
  98.  
  99.     CONST IS_DIR = 1
  100.     CONST IS_FILE = 2
  101.     DIM flags AS LONG, file_size AS LONG, DirCount AS INTEGER, FileCount AS INTEGER, length AS LONG
  102.     DIM nam$
  103.     REDIM _PRESERVE DirList(100), FileList(100)
  104.     DirCount = 0: FileCount = 0
  105.  
  106.     IF load_dir(SearchDirectory + CHR$(0)) THEN
  107.         DO
  108.             length = has_next_entry
  109.             IF length > -1 THEN
  110.                 nam$ = SPACE$(length)
  111.                 get_next_entry nam$, flags, file_size
  112.                 IF (flags AND IS_DIR) THEN
  113.                     DirCount = DirCount + 1
  114.                     IF DirCount > UBOUND(DirList) THEN REDIM _PRESERVE DirList(UBOUND(DirList) + 100)
  115.                     DirList(DirCount) = nam$
  116.                 ELSEIF (flags AND IS_FILE) THEN
  117.                     FileCount = FileCount + 1
  118.                     IF FileCount > UBOUND(filelist) THEN REDIM _PRESERVE FileList(UBOUND(filelist) + 100)
  119.                     FileList(FileCount) = nam$
  120.                 END IF
  121.             END IF
  122.         LOOP UNTIL length = -1
  123.         'close_dir 'move to after end if  might correct the multi calls problem
  124.     ELSE
  125.     END IF
  126.     close_dir 'this  might correct the multi calls problem
  127.  
  128.     REDIM _PRESERVE DirList(DirCount)
  129.     REDIM _PRESERVE FileList(FileCount)
  130.  
  131. FUNCTION rightOf$ (source$, of$)
  132.     IF INSTR(source$, of$) > 0 THEN rightOf$ = MID$(source$, INSTR(source$, of$) + LEN(of$))
  133.  
  134. ' "Escape or Red X box returns -1719 to allow a Cancel function and signal no slection."
  135. FUNCTION getArrayItemNumber& (locateRow, locateColumn, boxWidth, boxHeight, arr() AS STRING)
  136.     'Notes: locateRow, locateColumn for top right corner of selection box on screen in characters for LOCATE.
  137.     'boxWidth and boxHeight are in character units, again for locate and print at correct places.
  138.     'All displaying is restricted to inside the box, which has PgUP and PgDn as top and bottom lines in the display.
  139.  
  140.     DIM curRow AS INTEGER, curCol AS INTEGER, fg AS _UNSIGNED LONG, bg AS _UNSIGNED LONG
  141.     DIM maxWidth AS INTEGER, maxHeight AS INTEGER, page AS INTEGER, hlite AS INTEGER, mx AS INTEGER, my AS INTEGER
  142.     DIM lastMX AS INTEGER, lastMY AS INTEGER, row AS INTEGER, mb AS INTEGER
  143.     DIM lba AS LONG, uba AS LONG, choice AS LONG, kh AS LONG, index AS LONG
  144.     DIM clrStr AS STRING, b AS STRING, selNum&
  145.  
  146.     'save old settings to restore at end ofsub
  147.     curRow = CSRLIN
  148.     curCol = POS(0)
  149.     fg = _DEFAULTCOLOR
  150.     bg = _BACKGROUNDCOLOR
  151.     _KEYCLEAR
  152.  
  153.     maxWidth = boxWidth '       number of characters in box
  154.     maxHeight = boxHeight - 2 ' number of lines displayed of array at one time = 1 page
  155.     lba = LBOUND(arr)
  156.     uba = UBOUND(arr)
  157.     page = 0
  158.     hlite = 0 '                 line in display ready for selection by spacebar or if no number is started, enter
  159.     clrStr$ = SPACE$(maxWidth) 'clearing a display line
  160.  
  161.     GOSUB update '              show the beginning of the array items for selection
  162.  
  163.     'signal cancel selection process, exit sub with this unlikely index to signal canel
  164.     choice = -1719 'primes 7 and 8, not likely to be a select index of an array
  165.  
  166.     DO 'until get a selection or demand exit
  167.  
  168.         'handle the key stuff
  169.         kh& = _KEYHIT
  170.         IF kh& THEN
  171.             IF kh& > 0 AND kh& < 255 THEN
  172.                 IF INSTR("0123456789", CHR$(kh&)) > 0 THEN b$ = b$ + CHR$(kh&): GOSUB update
  173.                 IF CHR$(kh&) = "f" THEN
  174.                     'REDIM FILs(0) AS STRING     'hopefully this is already ready
  175.                     'loadFiles FILs()
  176.                     selNum& = getArrayItemNumber&(5, 60, 60, 30, FILs())
  177.                     COLOR _RGB32(180, 180, 255)
  178.                     CLS 'need to signal out of file selection
  179.                     IF selNum& >= LBOUND(FILs) AND selNum& <= UBOUND(FILs) THEN selectedFile = FILs(selNum&)
  180.                     EXIT DO
  181.                     'back to directory select
  182.                 END IF
  183.  
  184.                 IF CHR$(kh&) = "c" THEN b$ = "": GOSUB update
  185.                 IF kh& = 13 THEN 'enter pressed check if number is being entered?
  186.                     IF LEN(b$) THEN
  187.                         IF VAL(b$) >= lba AND VAL(b$) <= uba THEN 'we have number started
  188.                             choice = VAL(b$): EXIT DO
  189.                         ELSE 'clear b$ to show some response to enter
  190.                             b$ = "": GOSUB update 'clear the value that doesn't work
  191.                         END IF
  192.                     ELSE
  193.                         choice = hlite + page * maxHeight + lba 'must mean to select the highlighted item
  194.                     END IF
  195.                 END IF
  196.                 IF kh& = 27 THEN EXIT DO 'escape clause offered to Cancel selection process
  197.                 IF kh& = 32 THEN choice = hlite + page * maxHeight + lba 'best way to choose highlighted selection
  198.                 IF kh& = 8 THEN 'backspace to edit number
  199.                     IF LEN(b$) THEN b$ = LEFT$(b$, LEN(b$) - 1): GOSUB update
  200.                 END IF
  201.             ELSE
  202.                 SELECT CASE kh& 'choosing sections of array to display and highlighted item
  203.                     CASE 20736 'pg dn
  204.                         IF (page + 1) * maxHeight + lba <= uba THEN page = page + 1: GOSUB update
  205.                     CASE 18688 'pg up
  206.                         IF (page - 1) * maxHeight + lba >= lba THEN page = page - 1: GOSUB update
  207.                     CASE 18432 'up
  208.                         IF hlite - 1 < 0 THEN
  209.                             IF page > 0 THEN
  210.                                 page = page - 1: hlite = maxHeight - 1: GOSUB update
  211.                             END IF
  212.                         ELSE
  213.                             hlite = hlite - 1: GOSUB update
  214.                         END IF
  215.                     CASE 20480 'down
  216.                         IF (hlite + 1) + page * maxHeight + lba <= uba THEN 'ok to move up
  217.                             IF hlite + 1 > maxHeight - 1 THEN
  218.                                 page = page + 1: hlite = 0: GOSUB update
  219.                             ELSE
  220.                                 hlite = hlite + 1: GOSUB update
  221.                             END IF
  222.                         END IF
  223.                     CASE 18176 'home
  224.                         page = 0: hlite = 0: GOSUB update
  225.                     CASE 20224 ' end
  226.                         page = INT((uba - lba) / maxHeight): hlite = maxHeight - 1: GOSUB update
  227.                 END SELECT
  228.             END IF
  229.         END IF
  230.  
  231.         'handle the mouse stuff
  232.         WHILE _MOUSEINPUT
  233.             IF _MOUSEWHEEL = -1 THEN 'up?
  234.                 IF hlite - 1 < 0 THEN
  235.                     IF page > 0 THEN
  236.                         page = page - 1: hlite = maxHeight - 1: GOSUB update
  237.                     END IF
  238.                 ELSE
  239.                     hlite = hlite - 1: GOSUB update
  240.                 END IF
  241.             ELSEIF _MOUSEWHEEL = 1 THEN 'down?
  242.                 IF (hlite + 1) + page * maxHeight + lba <= uba THEN 'ok to move up
  243.                     IF hlite + 1 > maxHeight - 1 THEN
  244.                         page = page + 1: hlite = 0: GOSUB update
  245.                     ELSE
  246.                         hlite = hlite + 1: GOSUB update
  247.                     END IF
  248.                 END IF
  249.             END IF
  250.         WEND
  251.         mx = INT((_MOUSEX - locateColumn * 8) / 8) + 2: my = INT((_MOUSEY - locateRow * 16) / 16) + 2
  252.         IF _MOUSEBUTTON(1) THEN 'click contols or select array item
  253.             'clear mouse clicks
  254.             mb = _MOUSEBUTTON(1)
  255.             IF mb THEN 'clear it
  256.                 WHILE mb 'OK!
  257.                     IF _MOUSEINPUT THEN mb = _MOUSEBUTTON(1)
  258.                     _LIMIT 100
  259.                 WEND
  260.             END IF
  261.  
  262.             IF mx >= 1 AND mx <= maxWidth AND my >= 1 AND my <= maxHeight THEN
  263.                 choice = my + page * maxHeight + lba - 1 'select item clicked
  264.             ELSEIF mx >= 1 AND mx <= maxWidth AND my = 0 THEN 'page up or exit
  265.                 IF my = 0 AND (mx <= maxWidth AND mx >= maxWidth - 2) THEN 'exit sign
  266.                     EXIT DO 'escape plan for mouse click top right corner of display box
  267.                 ELSE 'PgUp bar clicked
  268.                     IF (page - 1) * maxHeight + lba >= lba THEN page = page - 1: GOSUB update
  269.                 END IF
  270.             ELSEIF mx >= 1 AND mx <= maxWidth AND my = maxHeight + 1 THEN 'page down bar clicked
  271.                 IF (page + 1) * maxHeight + lba <= uba THEN page = page + 1: GOSUB update
  272.             END IF
  273.         ELSE '   mouse over highlighting, only if mouse has moved!
  274.             IF mx >= 1 AND mx <= maxWidth AND my >= 1 AND my <= maxHeight THEN
  275.                 IF mx <> lastMX OR my <> lastMY THEN
  276.                     IF my - 1 <> hlite AND (my - 1 + page * maxHeight + lba <= uba) THEN
  277.                         hlite = my - 1
  278.                         lastMX = mx: lastMY = my
  279.                         GOSUB update
  280.                     END IF
  281.                 END IF
  282.             END IF
  283.         END IF
  284.         _LIMIT 200
  285.     LOOP UNTIL choice >= lba AND choice <= uba
  286.     getArrayItemNumber& = choice
  287.     COLOR fg, bg
  288.     'clear key presses
  289.     _KEYCLEAR
  290.     LOCATE curRow, curCol
  291.     'clear mouse clicks
  292.     mb = _MOUSEBUTTON(1)
  293.     IF mb THEN 'clear it
  294.         WHILE mb 'OK!
  295.             IF _MOUSEINPUT THEN mb = _MOUSEBUTTON(1)
  296.             _LIMIT 100
  297.         WEND
  298.     END IF
  299.     EXIT SUB
  300.  
  301.     update: '--------------- display of array sections and controls on screen
  302.  
  303.     'fix hlite if it has dropped below last array item
  304.     WHILE hlite + page * maxHeight + lba > uba
  305.         hlite = hlite - 1
  306.     WEND
  307.  
  308.     'main display of array items at page * maxHeight (lines high)
  309.     FOR row = 0 TO maxHeight - 1
  310.         IF hlite = row THEN COLOR _RGB(200, 200, 255), _RGB32(0, 0, 88) ELSE COLOR _RGB32(0, 0, 88), _RGB(200, 200, 255)
  311.         LOCATE locateRow + row, locateColumn: PRINT clrStr$
  312.         index = row + page * maxHeight + lba
  313.         IF index >= lba AND index <= uba THEN
  314.             LOCATE locateRow + row, locateColumn
  315.             PRINT LEFT$(LTRIM$(STR$(index)) + ") " + arr(index), maxWidth)
  316.         END IF
  317.     NEXT
  318.  
  319.     'make page up and down bars to click, print PgUp / PgDn if available
  320.     COLOR _RGB32(200, 200, 255), _RGB32(0, 100, 50)
  321.     LOCATE locateRow - 1, locateColumn: PRINT SPACE$(maxWidth)
  322.     IF page <> 0 THEN LOCATE locateRow - 1, locateColumn: PRINT LEFT$(" Pg Up" + SPACE$(maxWidth), maxWidth)
  323.     LOCATE locateRow + maxHeight, locateColumn: PRINT SPACE$(maxWidth)
  324.     IF page <> INT(uba / maxHeight) THEN
  325.         LOCATE locateRow + maxHeight, locateColumn: PRINT LEFT$(" Pg Dn" + SPACE$(maxWidth), maxWidth)
  326.     END IF
  327.     'make exit sign for mouse click
  328.     COLOR _RGB32(255, 255, 255), _RGB32(200, 100, 0)
  329.     LOCATE locateRow - 1, locateColumn + maxWidth - 3
  330.     PRINT " X "
  331.  
  332.     'if a number selection has been started show it's build = b$
  333.     IF LEN(b$) THEN
  334.         COLOR _RGB(255, 255, 0), _RGB32(0, 0, 0)
  335.         LOCATE locateRow + maxHeight, locateColumn + maxWidth - LEN(b$) - 1
  336.         PRINT b$;
  337.     END IF
  338.     _DISPLAY
  339.     _LIMIT 100
  340.     RETURN
  341.  
  342. ' for saving and restoring screen settins
  343. SUB scnState (restoreTF AS INTEGER) 'Thanks Steve McNeill
  344.     STATIC Font AS LONG, DefaultColor AS _UNSIGNED LONG, BackGroundColor AS _UNSIGNED LONG, Dest AS LONG, Source AS LONG
  345.     STATIC row AS INTEGER, col AS INTEGER, autodisplay AS INTEGER, mb AS INTEGER
  346.     IF restoreTF THEN
  347.         _FONT Font
  348.         COLOR DefaultColor, BackGroundColor
  349.         _DEST Dest
  350.         _SOURCE Source
  351.         LOCATE row, col
  352.         IF autodisplay THEN _AUTODISPLAY ELSE _DISPLAY
  353.         _KEYCLEAR
  354.         WHILE _MOUSEINPUT: WEND 'clear mouse clicks
  355.         mb = _MOUSEBUTTON(1)
  356.         IF mb THEN
  357.             DO
  358.                 WHILE _MOUSEINPUT: WEND
  359.                 mb = _MOUSEBUTTON(1)
  360.                 _LIMIT 100
  361.             LOOP UNTIL mb = 0
  362.         END IF
  363.     ELSE
  364.         Font = _FONT: DefaultColor = _DEFAULTCOLOR: BackGroundColor = _BACKGROUNDCOLOR
  365.         Dest = _DEST: Source = _SOURCE
  366.         row = CSRLIN: col = POS(0): autodisplay = _AUTODISPLAY
  367.     END IF
  368.  
  369.  

You will have to test the selected file for the proper extension.
Title: Re: BMP picture viewer (slide show)
Post by: NOVARSEG on February 05, 2021, 02:49:04 am
BMP slide show.  Copy the 4 attached files into a folder to make this work. (3 BMP images and 1 EXE)

Here is the source code BAS file

Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit"
  2.  
  3.  
  4.     FUNCTION load_dir& (s AS STRING)
  5.     FUNCTION has_next_entry& ()
  6.     SUB close_dir ()
  7.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  8.  
  9. DIM flags AS LONG
  10. DIM length AS LONG
  11. DIM F(100) AS STRING
  12. DIM Fcount AS INTEGER
  13. DIM tt AS STRING * 4
  14. DIM t AS STRING * 3
  15. DIM handle AS LONG
  16.  
  17. IF load_dir("." + CHR$(0)) = -1 THEN 'found "." this is a folder  not a root drive
  18.  
  19.     DO
  20.         length = has_next_entry
  21.         IF length = -1 THEN
  22.             '_FULLSCREEN
  23.             PRINT "NO more files to process"
  24.             EXIT DO
  25.         END IF
  26.  
  27.         IF length > -1 THEN
  28.             NAM$ = SPACE$(length)
  29.             get_next_entry NAM, flags, file_size
  30.  
  31.             IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  32.                 PRINT NAM + "                  "
  33.                 Fcount = Fcount + 1
  34.                 F(Fcount) = NAM
  35.             END IF
  36.  
  37.         END IF
  38.  
  39.     LOOP
  40.     Fmax = Fcount
  41.  
  42. END IF ''''IF load_dir("." + CHR$(0)) = -1 THEN
  43.  
  44. close_dir '
  45.  
  46.  
  47. Fcount = 0
  48.  
  49.     N = 0
  50.  
  51.     Fcount = Fcount + 1
  52.     OPEN F(Fcount) FOR BINARY AS #1
  53.     IF Fcount = Fmax THEN Fcount = 0
  54.  
  55.     GET #1, 11, OF
  56.     'PRINT "Offset to picture data"; OF
  57.  
  58.     GET #1, 15, L
  59.     ' PRINT "Header size "; L
  60.  
  61.     GET #1, , W
  62.     ' PRINT "image width "; W
  63.  
  64.     GET #1, , H
  65.     'PRINT "image height "; H
  66.  
  67.     GET #1, , I
  68.  
  69.  
  70.     GET #1, , I
  71.     ' PRINT "bits per pixel "; I
  72.     'PRINT
  73.  
  74.     IF I = 32 THEN: _FULLSCREEN: PRINT "Can't process 32 bit per pixel BMP": CLOSE: GOTO LL1
  75.     IF I <> 24 THEN PRINT "Not a 24 bit per pixel BMP": CLOSE: GOTO LL1
  76.  
  77.     DIM a AS _MEM
  78.     handle = _NEWIMAGE(W, H, 32)
  79.     a = _MEMIMAGE(handle)
  80.  
  81.  
  82.  
  83.     FOR R = H - 1 TO 0 STEP -1
  84.  
  85.         FOR C = 1 TO W * 3 STEP 3
  86.             GET #1, R * W * 3 + C + OF, t
  87.             tt = t + CHR$(255)
  88.             _MEMPUT a, a.OFFSET + N, tt
  89.             N = N + 4
  90.         NEXT C
  91.  
  92.     NEXT R
  93.  
  94.     SCREEN _NEWIMAGE(W, H, 32)
  95.     _PUTIMAGE (0, 0), handle
  96.     _MEMFREE a
  97.  
  98.     LL1:
  99.     t1 = TIMER
  100.     DO
  101.         _LIMIT 100
  102.         t2 = TIMER
  103.         IF t2 - t1 >= 2 THEN EXIT DO
  104.         i$ = INKEY$
  105.         IF i$ = CHR$(27) THEN END
  106.     LOOP
  107.     CLOSE
  108.  
  109.  

OK there is the pics hmm not much use for the program if you can see the pics ????










Title: Re: BMP picture viewer
Post by: Dav on February 05, 2021, 11:54:05 am
Works for me here.  Nice to see someone latch onto an programming idea and not stop till it gets done.

- Dav
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 01:21:18 pm
@NOVARSEG

Wow nice job re-interpreting the library for files/folders and using direntry.h !

And those 3 photos are gorgeous!

Thanks for providing files to test, now working, I made some mods you might like for slide show app:

Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit" ' got it to display can I get it to load
  2.  
  3.     FUNCTION load_dir& (s AS STRING)
  4.     FUNCTION has_next_entry& ()
  5.     SUB close_dir ()
  6.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  7.  
  8. DIM flags AS LONG
  9. DIM length AS LONG
  10. DIM F(100) AS STRING
  11. DIM Fcount AS INTEGER
  12. IF load_dir("." + CHR$(0)) = -1 THEN 'found "." this is a folder  not a root drive
  13.     DO
  14.         length = has_next_entry
  15.         IF length = -1 THEN
  16.             EXIT DO
  17.         END IF
  18.         IF length > -1 THEN
  19.             NAM$ = SPACE$(length)
  20.             get_next_entry NAM, flags, file_size
  21.             IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  22.                 PRINT NAM + "                  "
  23.                 Fcount = Fcount + 1
  24.                 F(Fcount) = NAM
  25.             END IF
  26.         END IF
  27.     LOOP
  28.     Fmax = Fcount
  29. END IF ''''IF load_dir("." + CHR$(0)) = -1 THEN
  30. close_dir '
  31.  
  32. SCREEN _NEWIMAGE(1240, 740, 32) ' as big as I can go for my laptop
  33. _DELAY .25
  34. COLOR , &HFF000033
  35. Fcount = 0
  36.     Fcount = Fcount + 1
  37.     image& = loadBMP24&(F(Fcount))
  38.     IF image& <> 0 THEN
  39.         _TITLE F(Fcount) + " press any for next image..."
  40.         _PUTIMAGE ((_WIDTH - _WIDTH(image&)) / 2, (_HEIGHT - _HEIGHT(image&)) / 2)-STEP(_WIDTH(image&), _HEIGHT(image&)), image&, 0
  41.         IF Fcount = Fmax THEN Fcount = 0
  42.         SLEEP
  43.         _FREEIMAGE image&
  44.         CLS
  45.     END IF
  46.  
  47. FUNCTION loadBMP24& (Filename$)
  48.     DIM OF AS _UNSIGNED LONG
  49.     DIM a AS _MEM
  50.     DIM handle AS LONG
  51.     DIM dh AS LONG
  52.     DIM tt AS STRING * 4
  53.     DIM t AS STRING * 3
  54.     IF RIGHT$(UCASE$(Filename$), 4) <> ".BMP" THEN EXIT FUNCTION
  55.     OPEN Filename$ FOR BINARY AS #1
  56.     GET #1, 11, OF
  57.     GET #1, 15, L
  58.     GET #1, , W
  59.     GET #1, , H
  60.     GET #1, , I
  61.     GET #1, , I 'get I twice ???
  62.     IF I <> 24 THEN EXIT FUNCTION
  63.     handle = _NEWIMAGE(W, H, 32)
  64.     a = _MEMIMAGE(handle)
  65.     FOR R = H - 1 TO 0 STEP -1
  66.         FOR C = 1 TO W * 3 STEP 3
  67.             GET #1, R * W * 3 + C + OF, t
  68.             tt = t + CHR$(255)
  69.             _MEMPUT a, a.OFFSET + N, tt
  70.             N = N + 4
  71.         NEXT C
  72.     NEXT R
  73.     CLOSE
  74.     dh = _NEWIMAGE(W, H, 32)
  75.     _PUTIMAGE (0, 0), handle, dh
  76.     loadBMP24& = dh
  77.     _MEMFREE a
  78.  

My main purpose was to get that FUNCTION working so we can explore the whole hard disk with Tiny Navigator File Retriever and maybe other image displaying routines or file management things.

EDIT: Oh I am going to need a check last 4 chars = BMP in that function because I won't be screening for that first as you had. Fixed.
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 01:45:54 pm
I see you have a question about getting I twice in your code, @bplus.  Here’s basically what you’re fetching, in the header of all BMP files:

Code: QB64: [Select]
  1. TYPE BMPEntry              ' Description                          Bytes    QB64 Function
  2.    ID AS STRING * 2        ' File ID("BM" text or 19778 AS Integer) 2      CVI("BM")
  3.    Size AS LONG            ' Total Size of the file                 4      LOF
  4.    Res1 AS INTEGER         ' Reserved 1 always 0                    2
  5.    Res2 AS INTEGER         ' Reserved 2 always 0                    2
  6.    Offset AS LONG          ' Start offset of image pixel data       4      (add one for GET)
  7. END TYPE                   '                                 Total 14
  8.  
  9. TYPE BMPHeader          'BMP header also used in Icon and Cursor files(.ICO and .CUR)
  10.    Hsize AS LONG           ' Info header size (always 40)           4
  11.    PWidth AS LONG          ' Image width                            4      _WIDTH(handle&)
  12.    PDepth AS LONG          ' Image height (doubled in icons)        4      _HEIGHT(handle&)
  13.    Planes AS INTEGER       ' Number of planes (normally 1)          2
  14.    BPP AS INTEGER          ' Bits per pixel(palette 1, 4, 8, 24)    2      _PIXELSIZE(handle&)
  15.    Compression AS LONG     ' Compression type(normally 0)           4
  16.    ImageBytes AS LONG      ' (Width + padder) * Height              4
  17.    Xres AS LONG            ' Width in PELS per metre(normally 0)    4
  18.    Yres AS LONG            ' Depth in PELS per metre(normally 0)    4
  19.    NumColors AS LONG       ' Number of Colors(normally 0)           4       2 ^ BPP
  20.    SigColors AS LONG       ' Significant Colors(normally 0)         4
  21. END TYPE                   '                 Total Header bytes =  40  

And if you want to know more about decoding them from there, let me know.  ;)

That first I you’re getting is for number of planes; second I is for bits per pixel.
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 01:50:42 pm
Thanks Steve, so that is how the 24 style BMP is detected.

Almost have Navigator ready to see if I have any  more of these special BMP's on my Hard Drive, I suspect JB images might be?
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 02:04:17 pm
One thing I’m curious about here:  How do you keep from warping your image, without accounting for the padding in the image data?

Bitmaps are strict on how many bits they store per row, and they need to be multiples of 4.  If your image doesnt fit that criteria, you add padding to it.

Say a 4 pixel wide image would be stored as RGBRGBRGBRGB.  12 bytes per row, no padding is needed.

But a 3 pixel wide image is RGBRGBRGB — or only 9 bytes.  You tack 3 bytes of padding onto the end of it.

From what I can tell, it doesn’t seem like there’s any adjusting for the padding in the code?  Will what you’re doing work with non-size-4 images?

EDIT:

IF ((BMP.PWidth * 3) MOD 4) <> 0 THEN        '’need padding pixels
    Padding$ = SPACE$((4 - ((BMP.PWidth * 3) MOD 4)))
END IF

EDIT 2:
You should get the padding at line 86.5 in your code, at least, that’s my initial thoughts on it.
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 02:18:55 pm
Yes now I can explore hard disk for BMP24 files and JB does use that format:
Code: QB64: [Select]
  1. ' B+ started 2019-08-22  restart 2020-06-24
  2. ' 2020-06-24 restart using DirEntry.h, wow looks like Steve's fix helped ALLOT! Thanks SMcNeill
  3. ' I am hitting glitches where Windows wont let me access I think.
  4. ' 2020-06-26 another fix to DirEntry.h usage add CHR$(0) to string when calling a C program.
  5. ' 2020-07-04 Use this to Kill some pesky .ogg files
  6. ' 2021-02-05 Use this to Display BMP24 files
  7.  
  8. ' direntry.h needs to be in QB64 folder ' <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< IMPORTANT
  9.     FUNCTION load_dir& (s AS STRING)
  10.     FUNCTION has_next_entry& ()
  11.     SUB close_dir ()
  12.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  13.  
  14. ' for GetFileName$()
  15. DIM SHARED selectedFile AS STRING
  16. REDIM SHARED DIRs(0) AS STRING, FILs(0) AS STRING
  17.  
  18. _TITLE "Tiny Navigator File Name Retriever for Displaying BMP24 files:  press f for File Selection of current directory"
  19. SCREEN _NEWIMAGE(1280, 740, 32)
  20. _DELAY .25
  21. _SCREENMOVE 65, 0 ' your screen might need different
  22.  
  23. DIM myFile$, image&
  24. COLOR , &HFF000033
  25. DO 'test our new function
  26.     scnState 0
  27.     myFile$ = GetFileName$
  28.     scnState -1 ' <<<<<<<<<<<<<<<<<<<<<<<<<<< this is supposed to restore back color
  29.     image& = loadBMP24&(myFile$)
  30.     COLOR , &HFF000033
  31.     CLS
  32.     IF image& <> 0 THEN
  33.         _TITLE myFile$ + " press any to get another BMP24 file image, esc to quit"  
  34.         _PUTIMAGE ((_WIDTH - _WIDTH(image&)) / 2, (_HEIGHT - _HEIGHT(image&)) / 2)-STEP(_WIDTH(image&), _HEIGHT(image&)), image&, 0
  35.         SLEEP
  36.         _FREEIMAGE image&
  37.     END IF
  38.  
  39. FUNCTION loadBMP24& (Filename$)
  40.     DIM OF AS _UNSIGNED LONG
  41.     DIM a AS _MEM
  42.     DIM handle AS LONG
  43.     DIM dh AS LONG
  44.     DIM tt AS STRING * 4
  45.     DIM t AS STRING * 3
  46.     IF RIGHT$(UCASE$(Filename$), 4) <> ".BMP" THEN EXIT FUNCTION
  47.     OPEN Filename$ FOR BINARY AS #1
  48.     GET #1, 11, OF
  49.     GET #1, 15, L
  50.     GET #1, , W
  51.     GET #1, , H
  52.     GET #1, , I
  53.     GET #1, , I 'get I twice ???
  54.     IF I <> 24 THEN EXIT FUNCTION
  55.     handle = _NEWIMAGE(W, H, 32)
  56.     a = _MEMIMAGE(handle)
  57.     FOR R = H - 1 TO 0 STEP -1
  58.         FOR C = 1 TO W * 3 STEP 3
  59.             GET #1, R * W * 3 + C + OF, t
  60.             tt = t + CHR$(255)
  61.             _MEMPUT a, a.OFFSET + N, tt
  62.             N = N + 4
  63.         NEXT C
  64.     NEXT R
  65.     CLOSE
  66.     dh = _NEWIMAGE(W, H, 32)
  67.     _PUTIMAGE (0, 0), handle, dh
  68.     loadBMP24& = dh
  69.     _MEMFREE a
  70.  
  71. FUNCTION GetFileName$ ()
  72.  
  73.     'you might try srnState before and after calling this function to restore conditions
  74.     ' uses: SUB GetLists (SearchDirectory AS STRING, DirList() AS STRING, FileList() AS STRING)  for getting dirs a nd files
  75.     ' uses: FUNCTION getArrayItemNumber& (locateRow, locateColumn, boxWidth, boxHeight, arr() AS STRING) for display large arrays and selecting
  76.  
  77.     ' share these with Main Code section:
  78.     ' 'for GetFileName$() and GetLists
  79.     'DIM SHARED selectedFile AS STRING
  80.     'REDIM SHARED DIRs(0) AS STRING, FILs(0) AS STRING
  81.  
  82.     DIM mySelection&, done$, i, t$
  83.  
  84.     _TITLE "Tiny Navigator File Name Retriever:  press f to start File Selection List of current directory"
  85.     DO
  86.         COLOR _RGB32(180, 180, 255), &HFF000000
  87.         CLS
  88.         t$ = "Current Directory: " + _CWD$
  89.         LOCATE 2, (_WIDTH / 8 - LEN(t$)) / 2: PRINT t$
  90.         REDIM DIRs(0) AS STRING, FILs(0) AS STRING
  91.         GetLists _CWD$, DIRs(), FILs()
  92.         FOR i = 0 TO UBOUND(FILs) ' this just offers a sample listing of files
  93.             IF i < 30 THEN LOCATE i + 4, 60: PRINT FILs(i) ELSE EXIT FOR
  94.         NEXT
  95.         mySelection& = getArrayItemNumber&(5, 5, 50, 30, DIRs())
  96.  
  97.         CLS
  98.         IF selectedFile <> "" THEN
  99.             GetFileName$ = selectedFile: done$ = "y"
  100.         ELSEIF mySelection& <> -1719 THEN
  101.             IF _TRIM$(DIRs(mySelection&)) <> "" THEN
  102.                 CHDIR DIRs(mySelection&)
  103.             END IF
  104.         ELSE
  105.             GetFileName$ = "": done$ = "y" ' cancelled or error
  106.         END IF
  107.         selectedFile = ""
  108.         _LIMIT 60
  109.     LOOP UNTIL done$ <> ""
  110.  
  111. SUB GetLists (SearchDirectory AS STRING, DirList() AS STRING, FileList() AS STRING)
  112.  
  113.     'put this block in main code section of your program close to top
  114.     '' direntry.h needs to be in QB64 folder
  115.     'DECLARE CUSTOMTYPE LIBRARY ".\direntry"
  116.     '    FUNCTION load_dir& (s AS STRING)
  117.     '    FUNCTION has_next_entry& ()
  118.     '    SUB close_dir ()
  119.     '    SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  120.     'END DECLARE
  121.  
  122.  
  123.     CONST IS_DIR = 1
  124.     CONST IS_FILE = 2
  125.     DIM flags AS LONG, file_size AS LONG, DirCount AS INTEGER, FileCount AS INTEGER, length AS LONG
  126.     DIM nam$
  127.     REDIM _PRESERVE DirList(100), FileList(100)
  128.     DirCount = 0: FileCount = 0
  129.  
  130.     IF load_dir(SearchDirectory + CHR$(0)) THEN
  131.         DO
  132.             length = has_next_entry
  133.             IF length > -1 THEN
  134.                 nam$ = SPACE$(length)
  135.                 get_next_entry nam$, flags, file_size
  136.                 IF (flags AND IS_DIR) THEN
  137.                     DirCount = DirCount + 1
  138.                     IF DirCount > UBOUND(DirList) THEN REDIM _PRESERVE DirList(UBOUND(DirList) + 100)
  139.                     DirList(DirCount) = nam$
  140.                 ELSEIF (flags AND IS_FILE) THEN
  141.                     FileCount = FileCount + 1
  142.                     IF FileCount > UBOUND(filelist) THEN REDIM _PRESERVE FileList(UBOUND(filelist) + 100)
  143.                     FileList(FileCount) = nam$
  144.                 END IF
  145.             END IF
  146.         LOOP UNTIL length = -1
  147.         'close_dir 'move to after end if  might correct the multi calls problem
  148.     ELSE
  149.     END IF
  150.     close_dir 'this  might correct the multi calls problem
  151.  
  152.     REDIM _PRESERVE DirList(DirCount)
  153.     REDIM _PRESERVE FileList(FileCount)
  154.  
  155. FUNCTION rightOf$ (source$, of$)
  156.     IF INSTR(source$, of$) > 0 THEN rightOf$ = MID$(source$, INSTR(source$, of$) + LEN(of$))
  157.  
  158. ' "Escape or Red X box returns -1719 to allow a Cancel function and signal no slection."
  159. FUNCTION getArrayItemNumber& (locateRow, locateColumn, boxWidth, boxHeight, arr() AS STRING)
  160.     'Notes: locateRow, locateColumn for top right corner of selection box on screen in characters for LOCATE.
  161.     'boxWidth and boxHeight are in character units, again for locate and print at correct places.
  162.     'All displaying is restricted to inside the box, which has PgUP and PgDn as top and bottom lines in the display.
  163.  
  164.     DIM curRow AS INTEGER, curCol AS INTEGER, fg AS _UNSIGNED LONG, bg AS _UNSIGNED LONG
  165.     DIM maxWidth AS INTEGER, maxHeight AS INTEGER, page AS INTEGER, hlite AS INTEGER, mx AS INTEGER, my AS INTEGER
  166.     DIM lastMX AS INTEGER, lastMY AS INTEGER, row AS INTEGER, mb AS INTEGER
  167.     DIM lba AS LONG, uba AS LONG, choice AS LONG, kh AS LONG, index AS LONG
  168.     DIM clrStr AS STRING, b AS STRING, selNum&
  169.  
  170.     'save old settings to restore at end ofsub
  171.     curRow = CSRLIN
  172.     curCol = POS(0)
  173.     fg = _DEFAULTCOLOR
  174.     bg = _BACKGROUNDCOLOR
  175.     _KEYCLEAR
  176.  
  177.     maxWidth = boxWidth '       number of characters in box
  178.     maxHeight = boxHeight - 2 ' number of lines displayed of array at one time = 1 page
  179.     lba = LBOUND(arr)
  180.     uba = UBOUND(arr)
  181.     page = 0
  182.     hlite = 0 '                 line in display ready for selection by spacebar or if no number is started, enter
  183.     clrStr$ = SPACE$(maxWidth) 'clearing a display line
  184.  
  185.     GOSUB update '              show the beginning of the array items for selection
  186.  
  187.     'signal cancel selection process, exit sub with this unlikely index to signal canel
  188.     choice = -1719 'primes 7 and 8, not likely to be a select index of an array
  189.  
  190.     DO 'until get a selection or demand exit
  191.  
  192.         'handle the key stuff
  193.         kh& = _KEYHIT
  194.         IF kh& THEN
  195.             IF kh& > 0 AND kh& < 255 THEN
  196.                 IF INSTR("0123456789", CHR$(kh&)) > 0 THEN b$ = b$ + CHR$(kh&): GOSUB update
  197.                 IF CHR$(kh&) = "f" THEN
  198.                     'REDIM FILs(0) AS STRING     'hopefully this is already ready
  199.                     'loadFiles FILs()
  200.                     selNum& = getArrayItemNumber&(5, 60, 60, 30, FILs())
  201.                     COLOR _RGB32(180, 180, 255)
  202.                     CLS 'need to signal out of file selection
  203.                     IF selNum& >= LBOUND(FILs) AND selNum& <= UBOUND(FILs) THEN selectedFile = FILs(selNum&)
  204.                     EXIT DO
  205.                     'back to directory select
  206.                 END IF
  207.  
  208.                 IF CHR$(kh&) = "c" THEN b$ = "": GOSUB update
  209.                 IF kh& = 13 THEN 'enter pressed check if number is being entered?
  210.                     IF LEN(b$) THEN
  211.                         IF VAL(b$) >= lba AND VAL(b$) <= uba THEN 'we have number started
  212.                             choice = VAL(b$): EXIT DO
  213.                         ELSE 'clear b$ to show some response to enter
  214.                             b$ = "": GOSUB update 'clear the value that doesn't work
  215.                         END IF
  216.                     ELSE
  217.                         choice = hlite + page * maxHeight + lba 'must mean to select the highlighted item
  218.                     END IF
  219.                 END IF
  220.                 IF kh& = 27 THEN EXIT DO 'escape clause offered to Cancel selection process
  221.                 IF kh& = 32 THEN choice = hlite + page * maxHeight + lba 'best way to choose highlighted selection
  222.                 IF kh& = 8 THEN 'backspace to edit number
  223.                     IF LEN(b$) THEN b$ = LEFT$(b$, LEN(b$) - 1): GOSUB update
  224.                 END IF
  225.             ELSE
  226.                 SELECT CASE kh& 'choosing sections of array to display and highlighted item
  227.                     CASE 20736 'pg dn
  228.                         IF (page + 1) * maxHeight + lba <= uba THEN page = page + 1: GOSUB update
  229.                     CASE 18688 'pg up
  230.                         IF (page - 1) * maxHeight + lba >= lba THEN page = page - 1: GOSUB update
  231.                     CASE 18432 'up
  232.                         IF hlite - 1 < 0 THEN
  233.                             IF page > 0 THEN
  234.                                 page = page - 1: hlite = maxHeight - 1: GOSUB update
  235.                             END IF
  236.                         ELSE
  237.                             hlite = hlite - 1: GOSUB update
  238.                         END IF
  239.                     CASE 20480 'down
  240.                         IF (hlite + 1) + page * maxHeight + lba <= uba THEN 'ok to move up
  241.                             IF hlite + 1 > maxHeight - 1 THEN
  242.                                 page = page + 1: hlite = 0: GOSUB update
  243.                             ELSE
  244.                                 hlite = hlite + 1: GOSUB update
  245.                             END IF
  246.                         END IF
  247.                     CASE 18176 'home
  248.                         page = 0: hlite = 0: GOSUB update
  249.                     CASE 20224 ' end
  250.                         page = INT((uba - lba) / maxHeight): hlite = maxHeight - 1: GOSUB update
  251.                 END SELECT
  252.             END IF
  253.         END IF
  254.  
  255.         'handle the mouse stuff
  256.         WHILE _MOUSEINPUT
  257.             IF _MOUSEWHEEL = -1 THEN 'up?
  258.                 IF hlite - 1 < 0 THEN
  259.                     IF page > 0 THEN
  260.                         page = page - 1: hlite = maxHeight - 1: GOSUB update
  261.                     END IF
  262.                 ELSE
  263.                     hlite = hlite - 1: GOSUB update
  264.                 END IF
  265.             ELSEIF _MOUSEWHEEL = 1 THEN 'down?
  266.                 IF (hlite + 1) + page * maxHeight + lba <= uba THEN 'ok to move up
  267.                     IF hlite + 1 > maxHeight - 1 THEN
  268.                         page = page + 1: hlite = 0: GOSUB update
  269.                     ELSE
  270.                         hlite = hlite + 1: GOSUB update
  271.                     END IF
  272.                 END IF
  273.             END IF
  274.         WEND
  275.         mx = INT((_MOUSEX - locateColumn * 8) / 8) + 2: my = INT((_MOUSEY - locateRow * 16) / 16) + 2
  276.         IF _MOUSEBUTTON(1) THEN 'click contols or select array item
  277.             'clear mouse clicks
  278.             mb = _MOUSEBUTTON(1)
  279.             IF mb THEN 'clear it
  280.                 WHILE mb 'OK!
  281.                     IF _MOUSEINPUT THEN mb = _MOUSEBUTTON(1)
  282.                     _LIMIT 100
  283.                 WEND
  284.             END IF
  285.  
  286.             IF mx >= 1 AND mx <= maxWidth AND my >= 1 AND my <= maxHeight THEN
  287.                 choice = my + page * maxHeight + lba - 1 'select item clicked
  288.             ELSEIF mx >= 1 AND mx <= maxWidth AND my = 0 THEN 'page up or exit
  289.                 IF my = 0 AND (mx <= maxWidth AND mx >= maxWidth - 2) THEN 'exit sign
  290.                     EXIT DO 'escape plan for mouse click top right corner of display box
  291.                 ELSE 'PgUp bar clicked
  292.                     IF (page - 1) * maxHeight + lba >= lba THEN page = page - 1: GOSUB update
  293.                 END IF
  294.             ELSEIF mx >= 1 AND mx <= maxWidth AND my = maxHeight + 1 THEN 'page down bar clicked
  295.                 IF (page + 1) * maxHeight + lba <= uba THEN page = page + 1: GOSUB update
  296.             END IF
  297.         ELSE '   mouse over highlighting, only if mouse has moved!
  298.             IF mx >= 1 AND mx <= maxWidth AND my >= 1 AND my <= maxHeight THEN
  299.                 IF mx <> lastMX OR my <> lastMY THEN
  300.                     IF my - 1 <> hlite AND (my - 1 + page * maxHeight + lba <= uba) THEN
  301.                         hlite = my - 1
  302.                         lastMX = mx: lastMY = my
  303.                         GOSUB update
  304.                     END IF
  305.                 END IF
  306.             END IF
  307.         END IF
  308.         _LIMIT 200
  309.     LOOP UNTIL choice >= lba AND choice <= uba
  310.     getArrayItemNumber& = choice
  311.     COLOR fg, bg
  312.     'clear key presses
  313.     _KEYCLEAR
  314.     LOCATE curRow, curCol
  315.     'clear mouse clicks
  316.     mb = _MOUSEBUTTON(1)
  317.     IF mb THEN 'clear it
  318.         WHILE mb 'OK!
  319.             IF _MOUSEINPUT THEN mb = _MOUSEBUTTON(1)
  320.             _LIMIT 100
  321.         WEND
  322.     END IF
  323.     EXIT SUB
  324.  
  325.     update: '--------------- display of array sections and controls on screen
  326.  
  327.     'fix hlite if it has dropped below last array item
  328.     WHILE hlite + page * maxHeight + lba > uba
  329.         hlite = hlite - 1
  330.     WEND
  331.  
  332.     'main display of array items at page * maxHeight (lines high)
  333.     FOR row = 0 TO maxHeight - 1
  334.         IF hlite = row THEN COLOR _RGB(200, 200, 255), _RGB32(0, 0, 88) ELSE COLOR _RGB32(0, 0, 88), _RGB(200, 200, 255)
  335.         LOCATE locateRow + row, locateColumn: PRINT clrStr$
  336.         index = row + page * maxHeight + lba
  337.         IF index >= lba AND index <= uba THEN
  338.             LOCATE locateRow + row, locateColumn
  339.             PRINT LEFT$(LTRIM$(STR$(index)) + ") " + arr(index), maxWidth)
  340.         END IF
  341.     NEXT
  342.  
  343.     'make page up and down bars to click, print PgUp / PgDn if available
  344.     COLOR _RGB32(200, 200, 255), _RGB32(0, 100, 50)
  345.     LOCATE locateRow - 1, locateColumn: PRINT SPACE$(maxWidth)
  346.     IF page <> 0 THEN LOCATE locateRow - 1, locateColumn: PRINT LEFT$(" Pg Up" + SPACE$(maxWidth), maxWidth)
  347.     LOCATE locateRow + maxHeight, locateColumn: PRINT SPACE$(maxWidth)
  348.     IF page <> INT(uba / maxHeight) THEN
  349.         LOCATE locateRow + maxHeight, locateColumn: PRINT LEFT$(" Pg Dn" + SPACE$(maxWidth), maxWidth)
  350.     END IF
  351.     'make exit sign for mouse click
  352.     COLOR _RGB32(255, 255, 255), _RGB32(200, 100, 0)
  353.     LOCATE locateRow - 1, locateColumn + maxWidth - 3
  354.     PRINT " X "
  355.  
  356.     'if a number selection has been started show it's build = b$
  357.     IF LEN(b$) THEN
  358.         COLOR _RGB(255, 255, 0), _RGB32(0, 0, 0)
  359.         LOCATE locateRow + maxHeight, locateColumn + maxWidth - LEN(b$) - 1
  360.         PRINT b$;
  361.     END IF
  362.     _DISPLAY
  363.     _LIMIT 100
  364.     RETURN
  365.  
  366. ' for saving and restoring screen settins
  367. SUB scnState (restoreTF AS INTEGER) 'Thanks Steve McNeill
  368.     STATIC Font AS LONG, DefaultColor AS _UNSIGNED LONG, BackGroundColor AS _UNSIGNED LONG, Dest AS LONG, Source AS LONG
  369.     STATIC row AS INTEGER, col AS INTEGER, autodisplay AS INTEGER, mb AS INTEGER
  370.     IF restoreTF THEN
  371.         _FONT Font
  372.         COLOR DefaultColor, BackGroundColor
  373.         _DEST Dest
  374.         _SOURCE Source
  375.         LOCATE row, col
  376.         IF autodisplay THEN _AUTODISPLAY ELSE _DISPLAY
  377.         _KEYCLEAR
  378.         WHILE _MOUSEINPUT: WEND 'clear mouse clicks
  379.         mb = _MOUSEBUTTON(1)
  380.         IF mb THEN
  381.             DO
  382.                 WHILE _MOUSEINPUT: WEND
  383.                 mb = _MOUSEBUTTON(1)
  384.                 _LIMIT 100
  385.             LOOP UNTIL mb = 0
  386.         END IF
  387.     ELSE
  388.         Font = _FONT: DefaultColor = _DEFAULTCOLOR: BackGroundColor = _BACKGROUNDCOLOR
  389.         Dest = _DEST: Source = _SOURCE
  390.         row = CSRLIN: col = POS(0): autodisplay = _AUTODISPLAY
  391.     END IF
  392.  
  393.  

Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 02:21:58 pm
Just read my edit above.  I believe you may run into issues with padding.  ;)
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 02:31:03 pm
Just read my edit above.  I believe you may run into issues with padding.  ;)

I did read and he's using the 4 size only.

Do you have an image to demonstrate and test code and code fixes?
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 03:14:27 pm
I did read and he's using the 4 size only.

Do you have an image to demonstrate and test code and code fixes?


Grab the file below and give it a run.  It's only 638 pixels wide (non-4), so it requires end of line padding.

Here's a quick set of example code for you to study as well: 
Code: QB64: [Select]
  1. SCREEN _NEWIMAGE(1024, 800, 32)
  2. handle = loadBMP24("non4.bmp")
  3. _PUTIMAGE , handle
  4.  
  5. FUNCTION loadBMP24& (Filename$)
  6.     DIM OF AS _UNSIGNED LONG
  7.     DIM a AS _MEM
  8.     DIM handle AS LONG
  9.     DIM dh AS LONG
  10.     DIM tt AS STRING * 4
  11.     DIM t AS STRING * 3
  12.  
  13.     IF RIGHT$(UCASE$(Filename$), 4) <> ".BMP" THEN EXIT FUNCTION
  14.     PRINT Filename$
  15.     OPEN Filename$ FOR BINARY AS #1
  16.     GET #1, 11, OF
  17.     GET #1, 15, L
  18.     GET #1, , W
  19.     GET #1, , H
  20.     GET #1, , I
  21.     GET #1, , I 'get I twice ???
  22.     IF I <> 24 THEN EXIT FUNCTION
  23.     handle = _NEWIMAGE(W, H, 32)
  24.     a = _MEMIMAGE(handle)
  25.     IF ((W * 3) MOD 4) <> 0 THEN '’need padding pixels
  26.         Padding$ = SPACE$((4 - ((W * 3) MOD 4)))
  27.     END IF
  28.     'SEEK #1, OF 'go to the offset where the data starts
  29.     FOR R = H - 1 TO 0 STEP -1
  30.         FOR C = 1 TO W * 3 STEP 3
  31.             GET #1, R * W * 3 + C + OF, t
  32.             'GET #1, ,t 'get the data sequentially
  33.             tt = t + CHR$(255)
  34.             _MEMPUT a, a.OFFSET + N, tt
  35.             N = N + 4
  36.         NEXT C
  37.         'GET #1, , Padding$ 'Get the padding at the end of the line
  38.     NEXT R
  39.     CLOSE
  40.  
  41.     dh = _NEWIMAGE(W, H, 32)
  42.     _PUTIMAGE (0, 0), handle, dh
  43.     loadBMP24& = dh
  44.     _MEMFREE a
  45.  

Run it and you'll see the image distorted as heck.

Unremark out the lines I've got remarked and run it.  (Also remark out the current GET line of  GET #1, R * W * 3 + C + OF, t     , as we're replacing it with the one I have remarked.)

You should be able to easily see the difference.

Here's an image of what the program you have now displays, vs the original:

 

Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 03:39:36 pm
Well the alignment is good but stripe colors don't match (see attached)

Here is the FUNCTION I used with the modifications
Code: QB64: [Select]
  1. FUNCTION loadBMP24& (Filename$)
  2.     DIM OF AS _UNSIGNED LONG
  3.     DIM a AS _MEM
  4.     DIM handle AS LONG
  5.     DIM dh AS LONG
  6.     DIM tt AS STRING * 4
  7.     DIM t AS STRING * 3
  8.     DIM padding$
  9.     IF RIGHT$(UCASE$(Filename$), 4) <> ".BMP" THEN EXIT FUNCTION
  10.     'PRINT Filename$
  11.     OPEN Filename$ FOR BINARY AS #1
  12.     GET #1, 11, OF
  13.     GET #1, 15, L
  14.     GET #1, , W
  15.     GET #1, , H
  16.     GET #1, , I
  17.     GET #1, , I 'get I twice ???
  18.     IF I <> 24 THEN EXIT FUNCTION
  19.     handle = _NEWIMAGE(W, H, 32)
  20.     a = _MEMIMAGE(handle)
  21.     IF ((W * 3) MOD 4) <> 0 THEN '’need padding pixels
  22.         padding$ = SPACE$((4 - ((W * 3) MOD 4)))
  23.     END IF
  24.     SEEK #1, OF 'go to the offset where the data starts
  25.     FOR R = H - 1 TO 0 STEP -1
  26.         FOR C = 1 TO W * 3 STEP 3
  27.             'GET #1, R * W * 3 + C + OF, t
  28.             GET #1, , t 'get the data sequentially
  29.             tt = t + CHR$(255)
  30.             _MEMPUT a, a.OFFSET + N, tt
  31.             N = N + 4
  32.         NEXT C
  33.         GET #1, , padding$ 'Get the padding at the end of the line
  34.     NEXT R
  35.     CLOSE
  36.  
  37.     dh = _NEWIMAGE(W, H, 32)
  38.     _PUTIMAGE (0, 0), handle, dh
  39.     loadBMP24& = dh
  40.     _MEMFREE a
  41.  
  42.  

Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 04:03:46 pm
Sorry.   A couple of little goofs on my part.

One was the SEEK.... I forgot to add 1 since binary files start at byte 1 and not 0.

    SEEK #1, OF + 1

Second is the face that it seems BMP files are stored bottom up, not top-down.  Just one of those many things I used to know, that I have to run into a little BLIP to remind myself of.   

Change the SEEK like above, and run it.  You'll see your image reversed.  :P

The change the MEMPUT to where you put the data in from the end to the beginning, and you're good to go:

           _MEMPUT a, a.OFFSET + a.SIZE - N - 4, tt 




And I think that'll account for your padding, start the color at the proper  pixel to read the RGB right, and draw it in the proper order.
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 04:31:00 pm
I had to go back and look at my own little BMP Save routine, to try and sort this out.  Here's the relevant part of the code, where we read the image from memory to save to file:

        w& = _WIDTH(image&)
        DO
            y = y - 1: x = x1% - 1
            DO
                x = x + 1
                _MEMGET m, m.OFFSET + (w& * y + x) * 4, temp


y starts at the bottom of the screen, and moves up.
x starts at the left of the screen, and moves right.

I'm thinking what I posted above might mirror the image for you, accidently.  (Not that you'd notice in single color rows...)


Here's what I'm thinking you actually need to do this properly:

Code: QB64: [Select]
  1.     IF I <> 24 THEN EXIT FUNCTION
  2.     handle = _NEWIMAGE(W, H, 32)
  3.     a = _MEMIMAGE(handle)
  4.     IF ((W * 3) MOD 4) <> 0 THEN '’need padding pixels
  5.         Padding$ = SPACE$((4 - ((W * 3) MOD 4)))
  6.     END IF
  7.     SEEK #1, OF + 1 'go to the offset where the data starts
  8.     FOR R = H - 1 TO 0 STEP -1 'from the bottom to top
  9.         FOR C = 0 TO W - 1 'left to right
  10.             GET #1, , t 'get the data sequentially
  11.             tt = t + CHR$(255)
  12.             _MEMPUT a, a.OFFSET + (W * R + C) * 4, tt
  13.         NEXT C
  14.         GET #1, , Padding$ 'Get the padding at the end of the line
  15.     NEXT R
  16.     CLOSE
  17.  


Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 04:49:02 pm
Thank you everyone for the nice comments!

bplus

 
Quote
GET #1, , I
    GET #1, , I 'get I twice ???

The first I  read 2 bytes (DIM I AS INTEGER)  which is biPlanes and I did not need that info, so skipped it

Quote
biPlanes   2   Must be 1
biBitCount   2   Bits per pixel - 1, 4, 8, 16, 24, or 32

The second I is bits per pixel and the viewer works only with 24 bits per pixel.
****
Another thing, in my source code, my comment looks to be wrong as I tried the code in the root of drive C and it worked. In the past I was able to detect whether code was run in the root or not and I will have to dig up some old code to see how that was done.
Quote
IF load_dir("." + CHR$(0)) = -1 THEN 'found "." this is a folder  not a root drive
****
Steve

In my viewer code I did not run into a padding issue.   If there was padding at the end of a row (scan line) then for sure my code would not work.   from http://www.dragonwins.com/domains/getteched/bmp/bmpfileformat.htm  Look at subheading "Pixel Data"  I'm really not sure if that is correct.

Will look into this "padding thing"


The only "padding" I did was adding that 4th byte at the end of a 24 bit pixel because
Quote
handle = _NEWIMAGE(W, H, 32)

expects a 32 bit pixel

****
Dav
 Glad it worked on your computer!!!




Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 06:20:33 pm
So I did some more researche

from https://homepages.inf.ed.ac.uk/rbf/BOOKS/PHILLIPS/cips2ed.pdf

Quote
The BMP color table has four bytes in each color table entry. The bytes
are for the blue, green, and red color values. The fourth byte is padding and
is always zero. For a 256 gray shade image, the color table is 4x256 bytes
long. The blue, green, and red values equal one another.
The final part of the BMP file is the image data. The data is stored row
by row with padding on the end of each row. The padding ensures the image
rows are multiples of four. The four, just like in the color table, makes it
easier to read blocks and keep track of addresses.

Now I'm really confused. You could have any number of bytes per scan line as long as they are a multiple of 3 (24 bit BMP).  I don't see the reason for padding in a 24 bit BMP???

If there was in fact padding at the end of a 24 bit BMP scan line, then my viewer code would not work.

As the above quote says

Quote
makes it easier to read blocks and keep track of addresses.

HUH? file pointers have BYTE resolution

Addresses WHAT , all we need to know is the offset from start of file to pixel data.

Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 06:26:53 pm
It’s like I told you:

You do one row at a time, starting at bottom left, to bottom right.
At the end of that row, you have padding to make it a length divisible by 4.
Then you go up to the line above it and repeat, until you reach the top.


You don’t need to worry about the color table reference, as that’s for other color modes.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 06:30:31 pm
Ya but I did not have to adjust for "padding" in my code . If there are padding bytes at the end of a scan line then the padding byte don't contain pixel data.  The file pointer (GET, point, . . .) is supposed to point to pixel data only

In a, say, 1024 * 768  24 bit BMP there are exactly 1024 * 3 bytes per scan line. That is what my code is saying.  If there was actually padding bytes at the end of a scan line then surely I would have found them.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 07:14:53 pm
OK maybe all my pics were 1024 pixels wide and  . . . . . .another was 1072

1024 * 3 /4 = 768  = no padding
1072 * 3 /4  = 804 = no padding

imagine that
  update coming
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 07:18:09 pm
Ya but I did not have to adjust for "padding" in my code . If there are padding bytes at the end of a scan line then the padding byte don't contain pixel data.  The file pointer (GET, point, . . .) is supposed to point to pixel data only

In a, say, 1024 * 768  24 bit BMP there are exactly 1024 * 3 bytes per scan line. That is what my code is saying.  If there was actually padding bytes at the end of a scan line then surely I would have found them.

As I mentioned, the padding is only there in images whose width is not evenly divisible by four.

Most full screen images have no padding: 320, 480, 640, 720, 1024 are all evenly divisible by 4.  Where you normally see the padding is when folks take a screenshot of a potion of the screen.  (Say from 0,0 to 638, 372 — that 638 isn’t evenly divisible by 4 and thus has padding at the end of the line.)
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 07:19:26 pm
Steve 

code update coming
Title: Re: BMP picture viewer
Post by: STxAxTIC on February 05, 2021, 07:28:04 pm
You guys are using this thread like a chatroom. Not illegal, but you realize Discord would be much more efficient?

Steve, you're my only hope. Drag them in.
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 07:52:47 pm
You guys are using this thread like a chatroom. Not illegal, but you realize Discord would be much more efficient?

Steve, you're my only hope. Drag them in.

Somebody forgot definition of a forum.
Title: Re: BMP picture viewer
Post by: STxAxTIC on February 05, 2021, 07:54:29 pm
Yeah, I'm raising the question of *who*.
Title: Re: BMP picture viewer
Post by: SMcNeill on February 05, 2021, 08:21:12 pm
Discord is no good for sharing larger snippets of code.  Forums are much better places for stuff like that.

Here's the complete version of what you guys want:

Code: QB64: [Select]
  1. SCREEN _NEWIMAGE(1024, 800, 32)
  2. handle = loadBMP24("non4.bmp")
  3. _PUTIMAGE , handle
  4.  
  5. FUNCTION loadBMP24& (Filename$)
  6.     DIM OF AS _UNSIGNED LONG
  7.     DIM a AS _MEM
  8.     DIM handle AS LONG
  9.     DIM dh AS LONG
  10.     DIM tt AS STRING * 4
  11.     DIM t AS STRING * 3
  12.  
  13.     IF RIGHT$(UCASE$(Filename$), 4) <> ".BMP" THEN EXIT FUNCTION
  14.     PRINT Filename$
  15.     OPEN Filename$ FOR BINARY AS #1
  16.     GET #1, 11, OF
  17.     GET #1, 15, L
  18.     GET #1, , W
  19.     GET #1, , H
  20.     GET #1, , I
  21.     GET #1, , I 'get I twice ???
  22.     IF I <> 24 THEN EXIT FUNCTION
  23.     handle = _NEWIMAGE(W, H, 32)
  24.     a = _MEMIMAGE(handle)
  25.     IF ((W * 3) MOD 4) <> 0 THEN '’need padding pixels
  26.         Padding$ = SPACE$((4 - ((W * 3) MOD 4)))
  27.     END IF
  28.     SEEK #1, OF + 1 'go to the offset where the data starts
  29.     FOR R = H - 1 TO 0 STEP -1 'from the bottom to top
  30.         FOR C = 0 TO W - 1 'left to right
  31.             GET #1, , t 'get the data sequentially
  32.             tt = t + CHR$(255)
  33.             _MEMPUT a, a.OFFSET + (W * R + C) * 4, tt
  34.         NEXT C
  35.         GET #1, , Padding$ 'Get the padding at the end of the line
  36.     NEXT R
  37.     CLOSE
  38.  
  39.  
  40.     'dh = _NEWIMAGE(W, H, 32)
  41.     '_PUTIMAGE (0, 0), handle, dh
  42.  
  43.     loadBMP24& = handle
  44.     _MEMFREE a
  45.  
  46.  

I posted the changes needed above, but didn't actually plug it into the whole thing until now.  Sorry.  Today's my mom's 81st birthday, and I was enjoying the evening with her and the rest of the family for a change.

One thing which I think is also important to change here:

Why copy the image over to another image of the exact same size, then return the handle to that image, without ever freeing the first image?  You guys have a rather  serious memory leak in your code here, and it's much simpler to just pass the handle on directly, without worrying about the middleman dh.  See the changes down at the last few lines in the routine, before you exit.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 05, 2021, 08:52:06 pm
Happy 81 Steve's Mom!

There is a lot in this thread to go over.  So I converted one of my 1024 pixel wide pics to 1023 pixels wide and my code crashed. Sorry about that.
Title: Re: BMP picture viewer
Post by: bplus on February 05, 2021, 09:07:39 pm
Quote
Why copy the image over to another image of the exact same size, then return the handle to that image, without ever freeing the first image?  You guys have a rather  serious memory leak in your code here, and it's much simpler to just pass the handle on directly, without worrying about the middleman dh.  See the changes down at the last few lines in the routine, before you exit.

Yeah that was me, just trying to get the thing to work as a Function returning a handle of a loaded image.

If this were a chat room I'd ask if that was a cobweb hanging off the bottom of the Starship model and I'd say congrats to Steve's mother and mention I took my 89 year old mother for COVID vaccination today and maybe mention nothing sticks in her memory so I have the same set of questions over and over... but if this were a chatroom you wouldn't have the image to check and say, "No, I think that is just a plain old hair."

I had that dang code working, went to clean up with the new Dim As _Unsigned Long all those variables into one line take out commented out code and when I was done, a blank screen.

So thanks Steve for writing up the revised sub because my cleanup refused to work right and it's even better without the extra handle and memory leaking code!

 

Title: Re: BMP picture viewer
Post by: NOVARSEG on February 06, 2021, 03:38:58 am
Some new code
 3 more pictures and an EXE to download

Fixed the padding bug.



Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit"
  2.  
  3.  
  4.     FUNCTION load_dir& (s AS STRING)
  5.     FUNCTION has_next_entry& ()
  6.     SUB close_dir ()
  7.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  8.  
  9. DIM flags AS LONG
  10. DIM length AS LONG
  11. DIM F(100) AS STRING
  12. DIM Fcount AS INTEGER
  13. DIM tt AS STRING * 4
  14. DIM t AS STRING * 3
  15. DIM handle AS LONG
  16.  
  17.  
  18. 'To view BMP pictures in the root drive or a particular folder (example)
  19. 'IF load_dir("C:" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  20. 'IF load_dir("C:\QB64" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  21.  
  22. 'To view BMP pictures in the directory (folder) from where the EXE is run (current)
  23. IF load_dir("." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  24.  
  25. 'To view BMP pictures in the parent directory (folder) from where the EXE is run (current)
  26. 'IF load_dir(".." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  27.  
  28.  
  29.     length = has_next_entry
  30.     IF length = -1 THEN
  31.         '_FULLSCREEN
  32.         PRINT "NO more files to process"
  33.         EXIT DO
  34.     END IF
  35.  
  36.     IF length > -1 THEN
  37.         NAM = SPACE$(length)
  38.         get_next_entry NAM, flags, file_size
  39.  
  40.  
  41.         IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  42.             PRINT NAM + "                  "
  43.             Fcount = Fcount + 1
  44.             F(Fcount) = NAM
  45.         END IF
  46.  
  47.     END IF
  48.  
  49. Fmax = Fcount
  50.  
  51. close_dir
  52.  
  53. Fcount = 0
  54.  
  55.     N = 0
  56.  
  57.     Fcount = Fcount + 1
  58.     OPEN F(Fcount) FOR BINARY AS #1
  59.  
  60.     GET #1, 11, OF
  61.     'PRINT "Offset to picture data"; OF
  62.  
  63.     GET #1, 15, L
  64.     ' PRINT "Header size "; L
  65.  
  66.     GET #1, , W
  67.     ' PRINT "image width "; W
  68.  
  69.     GET #1, , H
  70.     'PRINT "image height "; H
  71.  
  72.     GET #1, , I
  73.  
  74.     GET #1, , I 'bits per pixel. Must = 24 for the program to work.
  75.  
  76.     IF I <> 24 THEN
  77.         _FULLSCREEN
  78.         CLS
  79.         PRINT "width "; W; "    Height "; H; "    Bits per pixel "; I
  80.         PRINT Fcount
  81.         PRINT F(Fcount); " is not a 24 bit per pixel BMP"
  82.         DO
  83.             IF INKEY$ <> "" THEN _FULLSCREEN OFF: CLS: EXIT DO
  84.         LOOP
  85.         CLOSE: GOTO LL1
  86.     END IF
  87.  
  88.     DIM a AS _MEM
  89.     handle = _NEWIMAGE(W, H, 32)
  90.     a = _MEMIMAGE(handle)
  91.  
  92.     pad = 4 - (W * 3) MOD 4
  93.  
  94.     IF pad = 4 THEN pad = 0 'pad = multiple of 4
  95.  
  96.     FOR R = H - 1 TO 0 STEP -1
  97.  
  98.         FOR C = 1 TO W * 3 STEP 3
  99.             GET #1, R * (W * 3 + pad) + C + OF, t
  100.             tt = t + CHR$(255)
  101.             _MEMPUT a, a.OFFSET + N, tt
  102.             N = N + 4
  103.         NEXT C
  104.  
  105.     NEXT R
  106.  
  107.     SCREEN _NEWIMAGE(W, H, 32)
  108.     _PUTIMAGE (0, 0), handle
  109.     _MEMFREE a
  110.  
  111.  
  112.     t1 = TIMER
  113.     DO
  114.         _LIMIT 100
  115.         t2 = TIMER
  116.         IF t2 - t1 > 2 THEN EXIT DO
  117.         i$ = INKEY$
  118.         IF i$ = CHR$(27) THEN END
  119.     LOOP
  120.     LL1:
  121.  
  122.     CLOSE
  123.  
  124.     IF Fcount = Fmax THEN Fcount = 0
  125.  
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 06, 2021, 08:24:07 pm
So I will explain why this works

Quote
pad = 4 - (W * 3) MOD 4
 
    IF pad = 4 THEN pad = 0    'pad = multiple of 4
 
    FOR R = H - 1 TO 0 STEP -1
 
        FOR C = 1 TO W * 3 STEP 3
            GET #1, R * (W * 3 + pad) + C + OF, t
            tt = t + CHR$(255)
            _MEMPUT a, a.OFFSET + N, tt
            N = N + 4
        NEXT C
 
    NEXT R

FOR R = H - 1 TO 0 STEP -1

if say H = 768 then R is from 767 to 0  or 768 rows.

row 0  final
row 1
row 2
. . .
row 767   start


 The key thing here is the final row value R = 0 so
when R = 0


0 * (W * 3 + pad) + C + OF  = C + OF = the correct offset into row 0

There are (W * 3 + pad ) bytes per row. so when R = 1 then

 (W * 3 + pad) + C + OF  = the correct offset into row 1

with (W * 3 + pad) being the bytes in the preceding row 0

when R = 2 then

2 * (W * 3 + pad) + C + OF  = the correct offset into row 2
with  2 * (W * 3 + pad) = the total bytes of row 0 + row 1

and so on.

in reality the code down counts but the offset will still work out the same

Title: Re: BMP picture viewer
Post by: NOVARSEG on February 07, 2021, 12:01:47 am
Working on next update

from   https://stackoverflow.com/questions/9296059/read-pixel-value-in-bmp-file

Quote
Some bitmaps can be written with a negative height, so when you try to allocate your image data buffer, your code will crash with std::bad_alloc. Bitmaps with negative height means that the image data is stored top to bottom instead of the traditional bottom to top. Therefore, a slightly better version of the top level answer is (still not including portability for systems with different endianness and size of bytes):

Title: Re: BMP picture viewer
Post by: NOVARSEG on February 07, 2021, 02:55:13 am
Code update

Works with negative height BMPs too.

EXE attached.

Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit"
  2.  
  3.  
  4.     FUNCTION load_dir& (s AS STRING)
  5.     FUNCTION has_next_entry& ()
  6.     SUB close_dir ()
  7.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  8.  
  9. DIM flags AS LONG
  10. DIM length AS LONG
  11. DIM F(100) AS STRING
  12. DIM Fcount AS INTEGER
  13. DIM tt AS STRING * 4
  14. DIM t AS STRING * 3
  15. DIM handle AS LONG
  16.  
  17.  
  18. 'To view BMP pictures in the root drive or a particular folder (example)
  19. 'IF load_dir("C:" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  20. 'IF load_dir("C:\QB64" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  21.  
  22. 'To view BMP pictures in the directory (folder) from where the EXE is run (current)
  23. IF load_dir("." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  24.  
  25. 'To view BMP pictures in the parent directory (folder) from where the EXE is run (current)
  26. 'IF load_dir(".." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  27.  
  28.  
  29.     length = has_next_entry
  30.     IF length = -1 THEN
  31.         PRINT "No more files to process"
  32.         EXIT DO
  33.     END IF
  34.  
  35.     IF length > -1 THEN
  36.         NAM = SPACE$(length)
  37.         get_next_entry NAM, flags, file_size
  38.  
  39.         IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  40.             PRINT NAM + "                              "
  41.             Fcount = Fcount + 1
  42.             F(Fcount) = NAM
  43.         END IF
  44.  
  45.     END IF
  46.  
  47.  
  48. Fmax = Fcount
  49. close_dir
  50. Fcount = 0
  51.  
  52.     N = 0
  53.  
  54.     Fcount = Fcount + 1
  55.     OPEN F(Fcount) FOR BINARY AS #1
  56.  
  57.     GET #1, 11, OF
  58.     'PRINT "Offset to picture data"; OF
  59.  
  60.     GET #1, 15, L
  61.     'PRINT "Header size "; L
  62.  
  63.     GET #1, , W
  64.     'PRINT "image width "; W
  65.  
  66.     GET #1, , H
  67.     'PRINT "image height "; H
  68.  
  69.     GET #1, , I 'biPlanes. Specifies the number of color planes on the target device.
  70.  
  71.     GET #1, , I 'bits per pixel. Must = 24 for the program to work.
  72.  
  73.     IF I <> 24 THEN
  74.         CLS
  75.         PRINT "width "; W; "    Height "; H; "    Bits per pixel "; I
  76.         PRINT Fcount
  77.         PRINT F(Fcount); " is not a 24 bit per pixel BMP"
  78.         DO
  79.             IF INKEY$ <> "" THEN CLS: EXIT DO
  80.         LOOP
  81.         GOTO LL1
  82.     END IF
  83.  
  84.     DIM a AS _MEM
  85.     handle = _NEWIMAGE(W, ABS(H), 32)
  86.     a = _MEMIMAGE(handle)
  87.  
  88.     pad = 4 - (W * 3) MOD 4
  89.  
  90.     IF pad = 4 THEN pad = 0 'pad = multiple of 4
  91.  
  92.     IF H < 0 THEN
  93.  
  94.         FOR R = 0 TO ABS(H) - 1
  95.  
  96.             FOR C = 1 TO W * 3 STEP 3
  97.                 GET #1, R * (W * 3 + pad) + C + OF, t
  98.                 tt = t + CHR$(255)
  99.                 _MEMPUT a, a.OFFSET + N, tt
  100.                 N = N + 4
  101.             NEXT C
  102.  
  103.         NEXT R
  104.  
  105.     END IF
  106.  
  107.  
  108.     IF H > 0 THEN
  109.  
  110.         FOR R = H - 1 TO 0 STEP -1
  111.  
  112.             FOR C = 1 TO W * 3 STEP 3
  113.                 GET #1, R * (W * 3 + pad) + C + OF, t
  114.                 tt = t + CHR$(255)
  115.                 _MEMPUT a, a.OFFSET + N, tt
  116.                 N = N + 4
  117.             NEXT C
  118.  
  119.         NEXT R
  120.  
  121.     END IF
  122.  
  123.  
  124.     SCREEN _NEWIMAGE(W, (ABS(H)), 32)
  125.     _PUTIMAGE (0, 0), handle
  126.     _MEMFREE a
  127.  
  128.  
  129.     t1 = TIMER
  130.     DO
  131.         _LIMIT 100
  132.         t2 = TIMER
  133.         IF t2 - t1 > 2 THEN EXIT DO
  134.         i$ = INKEY$
  135.         IF i$ = CHR$(27) THEN END
  136.     LOOP
  137.     LL1:
  138.  
  139.     CLOSE
  140.  
  141.     IF Fcount = Fmax THEN Fcount = 0
  142.  
Title: Re: BMP picture viewer
Post by: _vince on February 07, 2021, 06:47:13 am
Nice, FYI this used to be my GOTO reference for whenever I needed to do anything with BMPs, explains the padding nonsense and some compression:
http://www.petesqbsite.com/sections/tutorials/zines/qbcm/12-bmps.html (http://www.petesqbsite.com/sections/tutorials/zines/qbcm/12-bmps.html)

Hey, Steve, are you fairly active on discord? Anyone know?
Title: Re: BMP picture viewer
Post by: SMcNeill on February 07, 2021, 07:19:12 am
Hey, Steve, are you fairly active on discord? Anyone know?

I go and come a lot.  The last week or so, I've been in fairly often, but for the next week or so, I probably won't be.  It's time for me to go get mom and take care of her again for a bit to give my sister a break.  ;)
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 07, 2021, 02:49:16 pm
_vince

RLE compression.  Another update

Steve

you were right about the padding. I should have listened.
Title: Re: BMP picture viewer
Post by: _vince on February 07, 2021, 06:56:01 pm
Yes, that padding nonsense is a common trap.  The code will work but will distort images with a certain width so it's important to test for.  The article mentions it I believe but I noticed that not all do.

I go and come a lot.  The last week or so, I've been in fairly often, but for the next week or so, I probably won't be.  It's time for me to go get mom and take care of her again for a bit to give my sister a break.  ;)

Nice, good to know! I'm going to make a discord account tonight and join in, see you soon ;-)
Title: Re: BMP picture viewer
Post by: bplus on February 09, 2021, 09:43:34 pm
To make a Sub of the BMP24, take the hunk of code between the 2 =========== bars
start this at bottom of code:

Code: QB64: [Select]
  1. Function LoadBmp24&(Filename$)  'loadBMP24& returns handle to image for display code from main code section.
  2. >>>>>>>>>>>>>>>> Insert Chuck Here
  3.     LoadBMP24& = handle

Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit"
  2.  
  3.  
  4.     FUNCTION load_dir& (s AS STRING)
  5.     FUNCTION has_next_entry& ()
  6.     SUB close_dir ()
  7.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  8.  
  9. DIM flags AS LONG
  10. DIM length AS LONG
  11. DIM F(100) AS STRING
  12. DIM Fcount AS INTEGER
  13. DIM tt AS STRING * 4
  14. DIM t AS STRING * 3
  15. DIM handle AS LONG
  16.  
  17.  
  18. 'To view BMP pictures in the root drive or a particular folder (example)
  19. 'IF load_dir("C:" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  20. 'IF load_dir("C:\QB64" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  21.  
  22. 'To view BMP pictures in the directory (folder) from where the EXE is run (current)
  23. IF load_dir("." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  24.  
  25. 'To view BMP pictures in the parent directory (folder) from where the EXE is run (current)
  26. 'IF load_dir(".." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  27.  
  28.  
  29.     length = has_next_entry
  30.     IF length = -1 THEN
  31.         PRINT "No more files to process"
  32.         EXIT DO
  33.     END IF
  34.  
  35.     IF length > -1 THEN
  36.         NAM = SPACE$(length)
  37.         get_next_entry NAM, flags, file_size
  38.  
  39.         IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  40.             PRINT NAM + "                              "
  41.             Fcount = Fcount + 1
  42.             F(Fcount) = NAM
  43.         END IF
  44.  
  45.     END IF
  46.  
  47.  
  48. Fmax = Fcount
  49. close_dir
  50. Fcount = 0
  51.  
  52.     N = 0
  53.  
  54.     Fcount = Fcount + 1
  55. '========================================================= start code chuck
  56.     OPEN F(Fcount) FOR BINARY AS #1   < change F(Fcount) to Sub parameter Filename$
  57.  
  58.     GET #1, 11, OF
  59.     'PRINT "Offset to picture data"; OF
  60.  
  61.     GET #1, 15, L
  62.     'PRINT "Header size "; L
  63.  
  64.     GET #1, , W
  65.     'PRINT "image width "; W
  66.  
  67.     GET #1, , H
  68.     'PRINT "image height "; H
  69.  
  70.     GET #1, , I 'biPlanes. Specifies the number of color planes on the target device.
  71.  
  72.     GET #1, , I 'bits per pixel. Must = 24 for the program to work.
  73.  
  74.     IF I <> 24 THEN
  75.         CLS
  76.         PRINT "width "; W; "    Height "; H; "    Bits per pixel "; I
  77.         PRINT Fcount
  78.         PRINT F(Fcount); " is not a 24 bit per pixel BMP"
  79.         DO
  80.             IF INKEY$ <> "" THEN CLS: EXIT DO
  81.         LOOP
  82.         GOTO LL1
  83.     END IF
  84.  
  85.     DIM a AS _MEM
  86.     handle = _NEWIMAGE(W, ABS(H), 32)
  87.     a = _MEMIMAGE(handle)
  88.  
  89.     pad = 4 - (W * 3) MOD 4
  90.  
  91.     IF pad = 4 THEN pad = 0 'pad = multiple of 4
  92.  
  93.     IF H < 0 THEN
  94.  
  95.         FOR R = 0 TO ABS(H) - 1
  96.  
  97.             FOR C = 1 TO W * 3 STEP 3
  98.                 GET #1, R * (W * 3 + pad) + C + OF, t
  99.                 tt = t + CHR$(255)
  100.                 _MEMPUT a, a.OFFSET + N, tt
  101.                 N = N + 4
  102.             NEXT C
  103.  
  104.         NEXT R
  105.  
  106.     END IF
  107.  
  108.  
  109.     IF H > 0 THEN
  110.  
  111.         FOR R = H - 1 TO 0 STEP -1
  112.  
  113.             FOR C = 1 TO W * 3 STEP 3
  114.                 GET #1, R * (W * 3 + pad) + C + OF, t
  115.                 tt = t + CHR$(255)
  116.                 _MEMPUT a, a.OFFSET + N, tt
  117.                 N = N + 4
  118.             NEXT C
  119.  
  120.         NEXT R
  121.  
  122.     END IF
  123. '===================================================== end of code chunk
  124.  
  125.     SCREEN _NEWIMAGE(W, (ABS(H)), 32)
  126.     _PUTIMAGE (0, 0), handle
  127.     _MEMFREE a
  128.  
  129.  
  130.     t1 = TIMER
  131.     DO
  132.         _LIMIT 100
  133.         t2 = TIMER
  134.         IF t2 - t1 > 2 THEN EXIT DO
  135.         i$ = INKEY$
  136.         IF i$ = CHR$(27) THEN END
  137.     LOOP
  138.     LL1:
  139.  
  140.     CLOSE
  141.  
  142.     IF Fcount = Fmax THEN Fcount = 0
  143.  
  144.  

Then move all DIM 's Variable in main that got moved to Function into Function at top
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 09, 2021, 10:11:19 pm
 
Code: QB64: [Select]
  1.  
  2.   'code  . . . . .
  3.  
  4. 'input filename
  5. 'returns W image width (if W = 0 then error)
  6. 'returns H image height (negative or positive value)
  7. 'returns HANDLE
  8. SUB loadBMP24 (FILENAME AS STRING, W AS LONG, H AS LONG, HANDLE AS LONG)
  9.     DIM OF AS _UNSIGNED LONG
  10.     DIM tt AS STRING * 4
  11.     DIM t AS STRING * 3
  12.     DIM pad AS INTEGER
  13.  
  14.     N = 0
  15.  
  16.     OPEN FILENAME FOR BINARY AS #1
  17.  
  18.     GET #1, 11, OF 'Offset to picture data
  19.  
  20.     GET #1, 15, L 'Header size
  21.  
  22.     GET #1, , W 'image width
  23.  
  24.     GET #1, , H 'image height
  25.  
  26.     GET #1, , I 'biPlanes. Specifies the number of color planes on the target device.
  27.  
  28.     GET #1, , I 'bits per pixel. Must = 24 for the program to work.
  29.  
  30.     IF I <> 24 THEN W = 0: GOTO LL1
  31.  
  32.     HANDLE = _NEWIMAGE(W, ABS(H), 32)
  33.     a = _MEMIMAGE(HANDLE)
  34.  
  35.     pad = 4 - (W * 3) MOD 4
  36.  
  37.     IF pad = 4 THEN pad = 0 'pad = multiple of 4
  38.  
  39.     IF H < 0 THEN
  40.  
  41.         FOR R = 0 TO ABS(H) - 1
  42.  
  43.             FOR C = 1 TO W * 3 STEP 3
  44.                 GET #1, R * (W * 3 + pad) + C + OF, t
  45.                 tt = t + CHR$(255)
  46.                 _MEMPUT a, a.OFFSET + N, tt
  47.                 N = N + 4
  48.             NEXT C
  49.  
  50.         NEXT R
  51.  
  52.     END IF
  53.  
  54.  
  55.     IF H > 0 THEN
  56.  
  57.         FOR R = H - 1 TO 0 STEP -1
  58.  
  59.             FOR C = 1 TO W * 3 STEP 3
  60.                 GET #1, R * (W * 3 + pad) + C + OF, t
  61.                 tt = t + CHR$(255)
  62.                 _MEMPUT a, a.OFFSET + N, tt
  63.                 N = N + 4
  64.             NEXT C
  65.  
  66.         NEXT R
  67.  
  68.     END IF
  69.     LL1:
  70.     CLOSE
Title: Re: BMP picture viewer
Post by: bplus on February 09, 2021, 10:48:23 pm
OK and now use the rest of the code for the main section to test calling up the Function or Sub.

I don't see the need to pass anything but the handle for the image to display in the main.

Do you have to change something when displaying for h < 0? Couldn't the adjustment be made before the sub or function passes the handle to main for display.

Remember to _FeeImage handles to avoid memory leaks.
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 09, 2021, 11:05:11 pm
@bplus

Quote
I don't see the need to pass anything but the handle for the image to display in the main.
   
In the main this code displays
SCREEN _NEWIMAGE(W, (ABS(H)), 32)
    _PUTIMAGE (0, 0), HANDLE
    _MEMFREE a

which requires W and H.  The SUB returns W, H and handle.  Notice the ABS(H) because the Sub will return either -H or H depending on the BMP

Also if there is an error (not a 24BMP) the sub will return W = 0

Maybe only a HANDLE is needed but at least there is the optional W and H returned

That is one of the problems with SUBs - what data do we think does not have to be returned
***
Quote
Do you have to change something when displaying for h < 0? Couldn't the adjustment be made before the sub or function passes the handle (height?) to main for display.

Inside the sub there could be an H = ABS(H) but what if your code in the main wanted to know if the BMP had -H or +h height?

****
Quote
Remember to _FreeImage handles to avoid memory leaks.

how  is _FreeImage used?
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 09, 2021, 11:43:45 pm
OK here is another version of the BMP viewer that shows how the SUB works

Code: QB64: [Select]
  1. _TITLE "BMP slide show.   Press ESC to exit"
  2.  
  3.  
  4.  
  5.     FUNCTION load_dir& (s AS STRING)
  6.     FUNCTION has_next_entry& ()
  7.     SUB close_dir ()
  8.     SUB get_next_entry (s AS STRING, flags AS LONG, file_size AS LONG)
  9.  
  10. DIM flags AS LONG
  11. DIM length AS LONG
  12. DIM F(100) AS STRING
  13. DIM Fcount AS INTEGER
  14. DIM tt AS STRING * 4
  15. DIM t AS STRING * 3
  16. DIM HANDLE AS LONG
  17.  
  18. 'To view BMP pictures in the root drive or a particular folder (example)
  19. 'IF load_dir("C:" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  20. 'IF load_dir("C:\QB64" + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  21.  
  22. 'To view BMP pictures in the directory (folder) from where the EXE is run (current)
  23. IF load_dir("." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  24.  
  25. 'To view BMP pictures in the parent directory (folder) from where the EXE is run (current)
  26. 'IF load_dir(".." + CHR$(0)) <> -1 THEN _FULLSCREEN: PRINT: PRINT "Path name does not exist": SLEEP: END
  27.  
  28.  
  29.     length = has_next_entry
  30.     IF length = -1 THEN
  31.         PRINT "No more files to process"
  32.         EXIT DO
  33.     END IF
  34.  
  35.     IF length > -1 THEN
  36.         NAM = SPACE$(length)
  37.         get_next_entry NAM, flags, file_size
  38.  
  39.         IF (flags XOR 2) = 0 AND UCASE$(RIGHT$(NAM, 3)) = "BMP" THEN
  40.             PRINT NAM + "                              "
  41.             Fcount = Fcount + 1
  42.             F(Fcount) = NAM
  43.         END IF
  44.  
  45.     END IF
  46.  
  47.  
  48. Fmax = Fcount
  49. close_dir
  50. Fcount = 0
  51.     Fcount = Fcount + 1
  52.  
  53.     loadBMP24 F(Fcount), W, H, HANDLE
  54.     IF W = 0 THEN
  55.         CLS
  56.         PRINT F(Fcount); " Is not a 24 bit per pixel BMP"
  57.         PRINT W; H
  58.         GOTO LL1
  59.     END IF
  60.  
  61.  
  62.  
  63.     SCREEN _NEWIMAGE(W, (ABS(H)), 32)
  64.     _PUTIMAGE (0, 0), HANDLE
  65.     _MEMFREE a
  66.  
  67.     LL1:
  68.     t1 = TIMER
  69.     DO
  70.         _LIMIT 100
  71.         t2 = TIMER
  72.         IF t2 - t1 > 1 THEN EXIT DO
  73.         i$ = INKEY$
  74.         IF i$ = CHR$(27) THEN END
  75.     LOOP
  76.  
  77.  
  78.     IF Fcount = Fmax THEN Fcount = 0
  79.  
  80.  
  81.  
  82.  
  83. 'SUB loadBMP24
  84. 'input filename
  85. 'returns W image width (if = 0 then error)
  86. 'returns H image height (negative or positive value)
  87. 'returns HANDLE
  88.  
  89. SUB loadBMP24 (FILENAME AS STRING, W AS LONG, H AS LONG, HANDLE AS LONG)
  90.  
  91.     DIM OF AS _UNSIGNED LONG
  92.     DIM tt AS STRING * 4
  93.     DIM t AS STRING * 3
  94.     DIM pad AS INTEGER
  95.  
  96.     N = 0
  97.  
  98.     OPEN FILENAME FOR BINARY AS #1
  99.  
  100.     GET #1, 11, OF 'Offset to picture data
  101.  
  102.     GET #1, 15, L 'Header size
  103.  
  104.     GET #1, , W 'image width
  105.  
  106.     GET #1, , H 'image height
  107.  
  108.     GET #1, , I 'biPlanes. Specifies the number of color planes on the target device.
  109.  
  110.     GET #1, , I 'bits per pixel. Must = 24 for the program to work.
  111.  
  112.     IF I <> 24 THEN W = 0: GOTO LL1
  113.  
  114.     HANDLE = _NEWIMAGE(W, ABS(H), 32)
  115.     a = _MEMIMAGE(HANDLE)
  116.  
  117.     pad = 4 - (W * 3) MOD 4
  118.  
  119.     IF pad = 4 THEN pad = 0 'pad = multiple of 4
  120.  
  121.     IF H < 0 THEN
  122.  
  123.         FOR R = 0 TO ABS(H) - 1
  124.  
  125.             FOR C = 1 TO W * 3 STEP 3
  126.                 GET #1, R * (W * 3 + pad) + C + OF, t
  127.                 tt = t + CHR$(255)
  128.                 _MEMPUT a, a.OFFSET + N, tt
  129.                 N = N + 4
  130.             NEXT C
  131.  
  132.         NEXT R
  133.  
  134.     END IF
  135.  
  136.  
  137.     IF H > 0 THEN
  138.  
  139.         FOR R = H - 1 TO 0 STEP -1
  140.  
  141.             FOR C = 1 TO W * 3 STEP 3
  142.                 GET #1, R * (W * 3 + pad) + C + OF, t
  143.                 tt = t + CHR$(255)
  144.                 _MEMPUT a, a.OFFSET + N, tt
  145.                 N = N + 4
  146.             NEXT C
  147.  
  148.         NEXT R
  149.  
  150.     END IF
  151.     LL1:
  152.     CLOSE
Title: Re: BMP picture viewer
Post by: bplus on February 10, 2021, 12:20:33 am
Works great! All the images loaded and displayed fine! Really easy to load the sub and use it for displaying images.

Navigate your hard disk for BMP24's to display:
Code: QB64: [Select]
  1. _Title "Pipecom Browser 2 for BMP24 File Display" 'b+ testing one array system with pipecom
  2. ' Thanks to Spriggsy, Zak, for convincing me pipecom is better!
  3. ' Using Steve's fixed BMP24 loader
  4. ' Also thanks to Dav for 1 window for file and folder selection idea
  5. ' Also thanks to NOVARSEG who got me going on this for a better filename$ retriever
  6. ' 2021-02-10 update with LoadBMP24 Sub
  7.  
  8. Declare Library "pipecom"
  9.     Function pipecom$ (cmd As String)
  10.  
  11. Screen _NewImage(1280, 740, 32)
  12. _Delay .25
  13. _ScreenMove 65, 0 ' your screen might need different
  14.  
  15. ReDim myFile$
  16. ReDim As Long col, row, charW, charH, W, H, image
  17. Color , &HFF000033
  18. Do 'test our new function
  19.     ScnState 0
  20.     col = 16: row = 3: charW = 128: charH = 40
  21.     myFile$ = GetFileName$(row, col, charW, charH)
  22.     ScnState -1 ' <<<<<<<<<<<<<<<<<<<<<<<<<<< this is supposed to restore back color
  23.     'image& = loadBMP24&(myFile$)
  24.  
  25.     loadBMP24 myFile$, W, H, image
  26.     Color , &HFF000033
  27.     Cls
  28.     If image <> 0 Then
  29.         _Title myFile$ + " press any to get another BMP24 file image, esc to quit"
  30.         _PutImage ((_Width - _Width(image&)) / 2, (_Height - _Height(image&)) / 2)-Step(_Width(image&), _Height(image&)), image&, 0
  31.         Sleep
  32.         _FreeImage image&
  33.     End If
  34.  
  35. 'NOVARSEG's fixed loadBLP24 Sub 2021-02-10 AAAAM https://www.qb64.org/forum/index.php?topic=3602.msg129753#msg129753
  36. Sub loadBMP24 (FILENAME As String, W As Long, H As Long, HANDLE As Long)
  37.  
  38.     Dim OF As _Unsigned Long
  39.     Dim tt As String * 4
  40.     Dim t As String * 3
  41.     Dim pad As Integer
  42.     Dim a As _MEM 'not defined
  43.  
  44.     N = 0
  45.  
  46.     Open FILENAME For Binary As #1
  47.     Get #1, 11, OF 'Offset to picture dat
  48.     Get #1, 15, L 'Header size
  49.     Get #1, , W 'image width
  50.     Get #1, , H 'image height
  51.     Get #1, , I 'biPlanes. Specifies the number of color planes on the target device.
  52.     Get #1, , I 'bits per pixel. Must = 24 for the program to work.
  53.     If I <> 24 Then W = 0: GoTo LL1
  54.     HANDLE = _NewImage(W, Abs(H), 32)
  55.     a = _MemImage(HANDLE)
  56.     pad = 4 - (W * 3) Mod 4
  57.     If pad = 4 Then pad = 0 'pad = multiple of 4
  58.     If H < 0 Then
  59.         For R = 0 To Abs(H) - 1
  60.             For C = 1 To W * 3 Step 3
  61.                 Get #1, R * (W * 3 + pad) + C + OF, t
  62.                 tt = t + Chr$(255)
  63.                 _MemPut a, a.OFFSET + N, tt
  64.                 N = N + 4
  65.             Next C
  66.         Next R
  67.     End If
  68.     If H > 0 Then
  69.         For R = H - 1 To 0 Step -1
  70.             For C = 1 To W * 3 Step 3
  71.                 Get #1, R * (W * 3 + pad) + C + OF, t
  72.                 tt = t + Chr$(255)
  73.                 _MemPut a, a.OFFSET + N, tt
  74.                 N = N + 4
  75.             Next C
  76.         Next R
  77.     End If
  78.     LL1:
  79.     Close
  80.  
  81. Function loadBMP24& (Filename$) ' Steve's revised version without the extra handle and memory leak
  82.     Dim As _Unsigned Long of, l, w, r, c
  83.     Dim a As _MEM
  84.     Dim As Long handle, h ' <<< edit as per NOVARSEG
  85.     Dim tt As String * 4
  86.     Dim t As String * 3
  87.     Dim padding$
  88.  
  89.     If Right$(UCase$(Filename$), 4) <> ".BMP" Then Exit Function
  90.     Print Filename$
  91.     Open Filename$ For Binary As #1
  92.     Get #1, 11, of
  93.     Get #1, 15, l
  94.     Get #1, , w
  95.     Get #1, , h
  96.     Get #1, , i
  97.     Get #1, , i 'get I twice ???
  98.     If i <> 24 Then Exit Function
  99.     handle = _NewImage(w, h, 32)
  100.     a = _MemImage(handle)
  101.     If ((w * 3) Mod 4) <> 0 Then padding$ = Space$((4 - ((w * 3) Mod 4))) '’need padding pixels
  102.     Seek #1, of + 1 'go to the offset where the data starts
  103.     For r = h - 1 To 0 Step -1 'from the bottom to top
  104.         For c = 0 To w - 1 'left to right
  105.             Get #1, , t 'get the data sequentially
  106.             tt = t + Chr$(255)
  107.             _MemPut a, a.OFFSET + (w * r + c) * 4, tt
  108.         Next c
  109.         Get #1, , padding$ 'Get the padding at the end of the line
  110.     Next r
  111.     Close
  112.     loadBMP24& = handle
  113.     _MemFree a
  114.  
  115. Function GetFileName$ (LocateR, LocateC, CharWide, CharHigh) ' < careful Locate Row, Col NOT x, y
  116.     'This Funtion needs:
  117.     ' pipecom.h in same folder as QB64.exe
  118.     ' This in main code section near top:
  119.     ' Declare Library "pipecom"
  120.     '    Function pipecom$ (cmd As String)
  121.     ' End Declare
  122.     ' sub  Split source$, delimiter$, arr$()
  123.     ' function GetArrayItem$(x, y, w, h, arr$())
  124.  
  125.     ReDim As String d, s, dr
  126.     ReDim As _Unsigned Long fc, bc
  127.     fc = _DefaultColor: bc = _BackgroundColor ' save colors
  128.     Getfolder:
  129.     Color fc, bc: Cls
  130.     d = pipecom("dir /n /o:gend") '/n files on right (40) /o = order g for group dirs first, e extension, name, date
  131.     d = Left$(d, Len(d) - 1) ' always ends with delimiter
  132.     ReDim dir(1 To 1) As String
  133.     Split d, Chr$(10), dir()
  134.     s = GetArrayItem$(LocateR, LocateC, CharWide, CharHigh, dir())
  135.     If InStr(s, "<DIR>") Then 'isolate name
  136.         dr = _Trim$(Mid$(s, InStr(s, "<DIR>") + 6))
  137.         ChDir dr
  138.         GoTo Getfolder
  139.     ElseIf InStr(s, "AM") Or InStr(s, "PM") Then
  140.         GetFileName$ = _Trim$(Mid$(s, 40))
  141.         Color fc, bc: Cls
  142.     ElseIf s = "" Then 'cancel, escape, bad line
  143.         Color fc, bc: Cls ' will return empty string
  144.     End If
  145.  
  146. Sub Split (SplitMeString As String, delim As String, loadMeArray() As String)
  147.     Dim curpos As Long, arrpos As Long, LD As Long, dpos As Long 'fix use the Lbound the array already has
  148.     curpos = 1: arrpos = LBound(loadMeArray): LD = Len(delim)
  149.     dpos = InStr(curpos, SplitMeString, delim)
  150.     Do Until dpos = 0
  151.         loadMeArray(arrpos) = Mid$(SplitMeString, curpos, dpos - curpos)
  152.         arrpos = arrpos + 1
  153.         If arrpos > UBound(loadMeArray) Then ReDim _Preserve loadMeArray(LBound(loadMeArray) To UBound(loadMeArray) + 1000) As String
  154.         curpos = dpos + LD
  155.         dpos = InStr(curpos, SplitMeString, delim)
  156.     Loop
  157.     loadMeArray(arrpos) = Mid$(SplitMeString, curpos)
  158.     ReDim _Preserve loadMeArray(LBound(loadMeArray) To arrpos) As String 'get the ubound correct
  159.  
  160.  
  161. ' Help: all this I hope is intuitive so Help window is provided
  162. ' "Mouse, mouse wheel, and arrow keys should work as expected for item selection."
  163. ' "Press spacebar to select a highlighted item or just click it."
  164. ' "Use number(s) + enter to select an array item by it's index number,"
  165. ' "backspace will remove last number pressed, delete will clear a number started.
  166. ' "Numbers started are shown in bottom right PgDn bar."
  167. ' "Enter will also select the highlighted item, if no number has been started."
  168. ' "Home starts you at lowest array index, End highlights then highest index."
  169. ' "Use PgUp and PgDn keys to flip through pages of array items."
  170. '
  171. ' Escape to Cancel Return "" else Return the selected string from the array
  172. Function GetArrayItem$ (LocateRoww, LocateColumn, BoxWidth, BoxHeight, Arr() As String)
  173.     'Notes: locateRow, locateColumn for top right corner of selection box on screen in characters for LOCATE.
  174.     'boxWidth and boxHeight are in character units, again for locate and print at correct places.
  175.     'All displaying is restricted to inside the box, which has PgUP and PgDn as top and bottom lines in the display.
  176.  
  177.     ReDim As Long maxWidth, maxHeight, page, hlite, mx, my, locateRow, lastMX, lastMY, row, mb
  178.     ReDim As Long lba, uba, choice, kh, index
  179.     Dim clrStr As String, b As String
  180.  
  181.     'ScnState 0 ' use out side this function before and after
  182.     locateRow = LocateRoww + 1 'fix a miscalc in coding
  183.     maxWidth = BoxWidth '       number of characters in box
  184.     maxHeight = BoxHeight - 2 ' number of lines displayed of array at one time = 1 page
  185.     lba = LBound(arr)
  186.     uba = UBound(arr)
  187.     page = 0
  188.     hlite = 0 '                 line in display ready for selection by spacebar or if no number is started, enter
  189.     clrStr$ = Space$(maxWidth) 'clearing a display line
  190.  
  191.     GoSub update '              show the beginning of the array items for selection
  192.     choice = -1719
  193.     Do 'until get a selection or demand exit
  194.  
  195.         'handle the key stuff
  196.         kh& = _KeyHit
  197.         If kh& Then
  198.             If kh& > 0 And kh& < 255 Then
  199.                 If InStr("0123456789", Chr$(kh&)) > 0 Then
  200.                     b$ = b$ + Chr$(kh&): GoSub update
  201.                 ElseIf kh& = 13 Then 'enter pressed check if number is being entered?
  202.                     If Len(b$) Then
  203.                         If Val(b$) >= lba And Val(b$) <= uba Then 'we have number started
  204.                             choice = Val(b$): Exit Do
  205.                         Else 'clear b$ to show some response to enter
  206.                             b$ = "": GoSub update 'clear the value that doesn't work
  207.                         End If
  208.                     Else
  209.                         choice = hlite + page * maxHeight + lba 'must mean to select the highlighted item
  210.                     End If
  211.                 ElseIf kh& = 27 Then
  212.                     Exit Do 'escape clause offered to Cancel selection process
  213.                 ElseIf kh& = 32 Then
  214.                     choice = hlite + page * maxHeight + lba 'best way to choose highlighted selection
  215.                 ElseIf kh& = 8 Then 'backspace to edit number
  216.                     If Len(b$) Then b$ = Left$(b$, Len(b$) - 1): GoSub update
  217.                 End If
  218.             Else
  219.                 Select Case kh& 'choosing sections of array to display and highlighted item
  220.                     Case 21248 ' delete so clear b$
  221.                         b$ = "": GoSub update
  222.                     Case 20736 ' pg dn
  223.                         If (page + 1) * maxHeight + lba <= uba Then page = page + 1: GoSub update
  224.                     Case 18688 ' pg up
  225.                         If (page - 1) * maxHeight + lba >= lba Then page = page - 1: GoSub update
  226.                     Case 18432 ' up
  227.                         If hlite - 1 < 0 Then
  228.                             If page > 0 Then
  229.                                 page = page - 1: hlite = maxHeight - 1: GoSub update
  230.                             End If
  231.                         Else
  232.                             hlite = hlite - 1: GoSub update
  233.                         End If
  234.                     Case 20480 'down
  235.                         If (hlite + 1) + page * maxHeight + lba <= uba Then 'ok to move up
  236.                             If hlite + 1 > maxHeight - 1 Then
  237.                                 page = page + 1: hlite = 0: GoSub update
  238.                             Else
  239.                                 hlite = hlite + 1: GoSub update
  240.                             End If
  241.                         End If
  242.                     Case 18176 'home
  243.                         page = 0: hlite = 0: GoSub update
  244.                     Case 20224 ' end
  245.                         page = Int((uba - lba) / maxHeight): hlite = maxHeight - 1: GoSub update
  246.                 End Select
  247.             End If
  248.         End If
  249.  
  250.         'handle the mouse stuff
  251.         While _MouseInput
  252.             If _MouseWheel = -1 Then 'up?
  253.                 If hlite - 1 < 0 Then
  254.                     If page > 0 Then
  255.                         page = page - 1: hlite = maxHeight - 1: GoSub update
  256.                     End If
  257.                 Else
  258.                     hlite = hlite - 1: GoSub update
  259.                 End If
  260.             ElseIf _MouseWheel = 1 Then 'down?
  261.                 If (hlite + 1) + page * maxHeight + lba <= uba Then 'ok to move up
  262.                     If hlite + 1 > maxHeight - 1 Then
  263.                         page = page + 1: hlite = 0: GoSub update
  264.                     Else
  265.                         hlite = hlite + 1: GoSub update
  266.                     End If
  267.                 End If
  268.             End If
  269.         Wend
  270.         mx = Int((_MouseX - LocateColumn * 8) / 8) + 2: my = Int((_MouseY - locateRow * 16) / 16) + 2
  271.         If _MouseButton(1) Then 'click contols or select array item
  272.             'clear mouse clicks
  273.             mb = _MouseButton(1)
  274.             If mb Then 'clear it
  275.                 While mb 'OK!
  276.                     If _MouseInput Then mb = _MouseButton(1)
  277.                     _Limit 100
  278.                 Wend
  279.             End If
  280.  
  281.             If mx >= 1 And mx <= maxWidth And my >= 1 And my <= maxHeight Then
  282.                 choice = my + page * maxHeight + lba - 1 'select item clicked
  283.             ElseIf mx >= 1 And mx <= maxWidth And my = 0 Then 'page up or exit
  284.                 If my = 0 And (mx <= maxWidth And mx >= maxWidth - 2) Then 'exit sign
  285.                     Exit Do 'escape plan for mouse click top right corner of display box
  286.                 Else 'PgUp bar clicked
  287.                     If (page - 1) * maxHeight + lba >= lba Then page = page - 1: GoSub update
  288.                 End If
  289.             ElseIf mx >= 1 And mx <= maxWidth And my = maxHeight + 1 Then 'page down bar clicked
  290.                 If (page + 1) * maxHeight + lba <= uba Then page = page + 1: GoSub update
  291.             End If
  292.         Else '   mouse over highlighting, only if mouse has moved!
  293.             If mx >= 1 And mx <= maxWidth And my >= 1 And my <= maxHeight Then
  294.                 If mx <> lastMX Or my <> lastMY Then
  295.                     If my - 1 <> hlite And (my - 1 + page * maxHeight + lba <= uba) Then
  296.                         hlite = my - 1
  297.                         lastMX = mx: lastMY = my
  298.                         GoSub update
  299.                     End If
  300.                 End If
  301.             End If
  302.         End If
  303.         _Limit 200
  304.     Loop Until choice >= lba And choice <= uba
  305.     If choice <> -1719 Then GetArrayItem$ = Arr(choice) 'set function and restore screen
  306.     'ScnState -1 'restore
  307.  
  308.     'display of array sections and controls on screen  ====================================================
  309.     update:
  310.  
  311.     'fix hlite if it has dropped below last array item
  312.     While hlite + page * maxHeight + lba > uba
  313.         hlite = hlite - 1
  314.     Wend
  315.  
  316.     'main display of array items at page * maxHeight (lines high)
  317.     For row = 0 To maxHeight - 1
  318.         If hlite = row Then Color _RGB(200, 200, 255), _RGB32(0, 0, 88) Else Color _RGB32(0, 0, 88), _RGB(200, 200, 255)
  319.         Locate locateRow + row, LocateColumn: Print clrStr$;
  320.         index = row + page * maxHeight + lba
  321.         If index >= lba And index <= uba Then
  322.             Locate locateRow + row, LocateColumn
  323.             Print Left$(LTrim$(Str$(index)) + ") " + Arr(index), maxWidth);
  324.         End If
  325.     Next
  326.  
  327.     'make page up and down bars to click, print PgUp / PgDn if available
  328.     Color _RGB32(200, 200, 255), _RGB32(0, 100, 50)
  329.     Locate locateRow - 1, LocateColumn: Print Space$(maxWidth);
  330.     If page <> 0 Then Locate locateRow - 1, LocateColumn: Print Left$(" Pg Up" + Space$(maxWidth), maxWidth);
  331.     Locate locateRow + maxHeight, LocateColumn: Print Space$(maxWidth);
  332.     If page <> Int(uba / maxHeight) Then
  333.         Locate locateRow + maxHeight, LocateColumn: Print Left$(" Pg Dn" + Space$(maxWidth), maxWidth);
  334.     End If
  335.     'make exit sign for mouse click
  336.     Color _RGB32(255, 255, 255), _RGB32(200, 100, 0)
  337.     Locate locateRow - 1, LocateColumn + maxWidth - 3
  338.     Print " X ";
  339.  
  340.     'if a number selection has been started show it's build = b$
  341.     If Len(b$) Then
  342.         Color _RGB(255, 255, 0), _RGB32(0, 0, 0)
  343.         Locate locateRow + maxHeight, LocateColumn + maxWidth - Len(b$) - 1
  344.         Print b$;
  345.     End If
  346.     _Display
  347.     _Limit 100
  348.     Return
  349.  
  350. Sub ScnState (restoreTF As Long) 'Thanks Steve McNeill  should we get a snap shot of screen?
  351.     Static As _Unsigned Long defaultColor, backGroundColor
  352.     Static As Long font, dest, source, row, col, autodisplay, mb, snap
  353.     If restoreTF Then
  354.         _Font font
  355.         Color defaultColor, backGroundColor
  356.         _Dest dest
  357.         _Source source
  358.  
  359.         _KeyClear
  360.         While _MouseInput: Wend 'clear mouse clicks
  361.         mb = _MouseButton(1)
  362.         If mb Then 'need this if line ?
  363.             Do
  364.                 While _MouseInput: Wend
  365.                 mb = _MouseButton(1)
  366.                 _Limit 100
  367.             Loop Until mb = 0
  368.         End If
  369.         _PutImage , snap, dest
  370.         _FreeImage snap
  371.         _Display
  372.         If autodisplay Then _AutoDisplay Else _Display
  373.         Locate row, col
  374.     Else
  375.         snap = _NewImage(_Width, _Height, 32)
  376.         _PutImage , 0, snap
  377.         font = _Font: defaultColor = _DefaultColor: backGroundColor = _BackgroundColor
  378.         dest = _Dest: source = _Source
  379.         row = CsrLin: col = Pos(0): autodisplay = _AutoDisplay
  380.         While _MouseInput: Wend 'clear mouse clicks
  381.         mb = _MouseButton(1)
  382.         If mb Then 'need this if line ?
  383.             Do
  384.                 While _MouseInput: Wend
  385.                 mb = _MouseButton(1)
  386.                 _Limit 100
  387.             Loop Until mb = 0
  388.         End If
  389.         _KeyClear
  390.     End If
  391.  
  392.  
  393.  
  394.  

Need this in your QB64.exe Folder
Title: Re: BMP picture viewer
Post by: NOVARSEG on February 10, 2021, 12:40:17 am
That is the first sub I have written in years!
 Do  I need the latest ver of QB64 to see this?   

Glad it worked even on -h too ?

i see you used  _FreeImage image&

Ok prob works better than  _MEMFREE a   
Title: Re: BMP picture viewer
Post by: bplus on February 10, 2021, 10:53:54 am
That is the first sub I have written in years!
 Do  I need the latest ver of QB64 to see this?   

Glad it worked even on -h too ?

i see you used  _FreeImage image&

Ok prob works better than  _MEMFREE a   

For your version QB64, you might have to change the REDIM statements that declare Types in groups like:
ReDim As Long x, y, z

Just make:
Redim x as long, y as long, z as long

Or get the 1.5 dev version ;)


I used _FreeImage because what was stored in Handle was an image.

_MEMFREE a should have been in LoadBMP24 Sub because that is where the variable, a, resides.
I don't know with _Mem stuff if a is automatically cleared like a regular variable as soon as you exit the sub. We know images wont.