QB64.org Forum

Active Forums => Programs => Topic started by: SMcNeill on November 19, 2019, 06:25:58 am

Title: Isometric Mapping Demo
Post by: SMcNeill on November 19, 2019, 06:25:58 am
I was thinking of working up a simple little game which utilizes isometric mapping to create a pseudo-3d environment, and I couldn't find a demo of anything similar on the forums.  So, in the spirit of "can do", I did, and here's what I came up with:

Code: QB64: [Select]
  1. SCREEN _NEWIMAGE(1024, 720, 32)
  2.  
  3. DIM SHARED GridSize AS INTEGER
  4. CONST Red = &HFFFF0000, Green = &HFF00FF00
  5. GridSize = 16 'My demo map is only going to be 16x16 pixels
  6.  
  7.  
  8. DIM grid(9, 9) AS _BYTE 'create a grid to layout our map
  9.  
  10. FOR i = 0 TO 9 'and let's create that map.  It's just going to be 4 walls and a floor...
  11.     grid(i, 0) = 1 'wall
  12.     grid(i, 9) = 1
  13.     grid(0, i) = 1
  14.     grid(9, i) = 1
  15.  
  16. FOR y = 1 TO 8
  17.     FOR x = 1 TO 8
  18.         grid(x, y) = 2 'floor
  19.     NEXT
  20.  
  21.  
  22. 'Now, let's just draw grids one at a time to compare.
  23. _PRINTSTRING (100, 80), "Normal 2D grid"
  24. FOR y = 0 TO 9
  25.     FOR x = 0 TO 9
  26.         'normal grid first
  27.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  28.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  29.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  30.         LINE (xpos, ypos)-(xpos2, ypos2), Kolor, BF
  31.         LINE (xpos, ypos)-(xpos2, ypos2), &HFFFFFFFF, B
  32.         'and that's all it takes for the normal grid.  Can't get any simpler for 2d maps!
  33.     NEXT
  34.  
  35.  
  36. _PRINTSTRING (350, 80), "Normal 2D grid in Isometic Perspective"
  37. FOR y = 0 TO 9
  38.     FOR x = 0 TO 9
  39.         'And now, let's do the same thing with our isometic map.
  40.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  41.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  42.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  43.         IsoLine xpos, ypos, xpos2, ypos2, 500, 0, Kolor
  44.         'And that's basically all the takes to rotate our map to make it a 2D isometic perspective
  45.     NEXT
  46.  
  47.  
  48. _PRINTSTRING (350, 360), "Normal 2D grid in 3D Isometic Perspective"
  49. FOR y = 0 TO 9
  50.     FOR x = 0 TO 9
  51.         'And here, I'm going to make it a 3D isometic map
  52.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  53.         IF grid(x, y) = 1 THEN z = 16 ELSE z = 0 'Give my walls a height of 16, for a cube
  54.  
  55.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  56.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  57.         IsoLine3D xpos, ypos, xpos2, ypos2, z, 500, 300, Kolor
  58.     NEXT
  59.  
  60.  
  61.  
  62.  
  63.  
  64.  
  65.  
  66.  
  67.  
  68. FUNCTION CX2I (x AS LONG, y AS LONG) 'Convert Cartesian X To Isometic coordinates
  69.     CX2I = x - y
  70.  
  71. FUNCTION CY2I (x AS LONG, y AS LONG) 'Convert Cartesian Y To Isometic coordinates
  72.     CY2I = (x + y) / 2
  73.  
  74. SUB IsoLine (x, y, x2, y2, xoffset, yoffset, kolor AS _UNSIGNED LONG)
  75.     'since we're drawing a diamond and not a square box, we can't use Line BF.
  76.     'We have to manually down the 4 points of the line.
  77.     LINE (CX2I(x, y) + xoffset, CY2I(x, y) + yoffset)-(CX2I(x2, y) + xoffset, CY2I(x2, y) + yoffset), kolor
  78.     LINE -(CX2I(x2, y2) + xoffset, CY2I(x2, y2) + yoffset), kolor
  79.     LINE -(CX2I(x, y2) + xoffset, CY2I(x, y2) + yoffset), kolor
  80.     LINE -(CX2I(x, y) + xoffset, CY2I(x, y) + yoffset), kolor
  81.     PAINT (CX2I(x, y) + xoffset, CY2I(x, y) + 4), kolor 'and fill the diamond solid
  82.     LINE (CX2I(x, y) + xoffset, CY2I(x, y) + yoffset)-(CX2I(x2, y) + xoffset, CY2I(x2, y) + yoffset), &HFFFFFFFF
  83.     LINE -(CX2I(x2, y2) + xoffset, CY2I(x2, y2) + yoffset), &HFFFFFFFF
  84.     LINE -(CX2I(x, y2) + xoffset, CY2I(x, y2) + yoffset), &HFFFFFFFF
  85.     LINE -(CX2I(x, y) + xoffset, CY2I(x, y) + yoffset), &HFFFFFFFF
  86.  
  87. SUB IsoLine3D (x, y, x2, y2, z, xoffset, yoffset, kolor AS _UNSIGNED LONG)
  88.     'Like IsoLine, we're going to have to draw our lines manually.
  89.     'only in this case, we also need a Z coordinate to tell us how THICK/TALL/HIGH to make our tile
  90.  
  91.     'Let's just do all the math first this time.
  92.     'We need to turn those 4 normal points into 4 isometric points (x, y, x1, y1)
  93.  
  94.     TempX1 = CX2I(x, y) + xoffset: TempY1 = CY2I(x, y) + yoffset
  95.     TempX2 = CX2I(x2, y) + xoffset: TempY2 = CY2I(x2, y) + yoffset
  96.     TempX3 = CX2I(x2, y2) + xoffset: TempY3 = CY2I(x2, y2) + yoffset
  97.     TempX4 = CX2I(x, y2) + xoffset: TempY4 = CY2I(x, y2) + yoffset
  98.  
  99.     'The top
  100.     LINE (TempX1, TempY1 - z)-(TempX2, TempY2 - z), kolor 'draw the diamond
  101.     LINE -(TempX3, TempY3 - z), kolor
  102.     LINE -(TempX4, TempY4 - z), kolor
  103.     LINE -(TempX1, TempY1 - z), kolor
  104.     PAINT (TempX1, TempY1 - z + 4), kolor 'and fill the diamond solid
  105.     LINE (TempX1, TempY1 - z)-(TempX2, TempY2 - z), -1 'and redraw the grid
  106.     LINE -(TempX3, TempY3 - z), -1
  107.     LINE -(TempX4, TempY4 - z), -1
  108.     LINE -(TempX1, TempY1 - z), -1
  109.  
  110.     IF z <> 0 THEN 'no need to draw any height, if there isn't any.
  111.         'the left side
  112.         LINE (TempX4, TempY4 - z)-(TempX4, TempY4), kolor 'draw it
  113.         LINE -(TempX3, TempY3), kolor
  114.         LINE -(TempX3, TempY3 - z), kolor
  115.         LINE -(TempX4, TempY4 - z), kolor
  116.         PAINT (TempX3 - 2, TempY3 - z + 2), kolor 'fill it
  117.         LINE (TempX4, TempY4 - z)-(TempX4, TempY4), -1 'redraw the grid lines
  118.         LINE -(TempX3, TempY3), -1
  119.         LINE -(TempX3, TempY3 - z), -1
  120.         LINE -(TempX4, TempY4 - z), -1
  121.         'and then for the right side
  122.         LINE (TempX3, TempY3 - z)-(TempX3, TempY3), kolor 'draw it
  123.         LINE -(TempX2, TempY2), kolor
  124.         LINE -(TempX2, TempY2 - z), kolor
  125.         LINE -(TempX3, TempY3 - z), kolor
  126.         PAINT (TempX3 + 2, TempY3 - z + 2), kolor 'fill it
  127.         LINE (TempX3, TempY3 - z)-(TempX3, TempY3), -1 'redraw the grid lines
  128.         LINE -(TempX2, TempY2), -1
  129.         LINE -(TempX2, TempY2 - z), -1
  130.         LINE -(TempX3, TempY3 - z), -1
  131.     END IF
  132.  
  133.  
  134.  


It's a simple enough little demo with nothing more than a green floor and red walls, but I think it highlights the process well enough for us.

My question for all you math folks out there: What's a simple little formula to use to get a paint point inside those boxes?  I really don't care for the hardcoded + 2 which I used, as it can paint the wrong things if the user draws a box too small for it.  There should be a simple, failsafe method which we can use that won't paint out of bounds, like this little demo does, but I'll be danged if I can come up with it at the moment.  It's too early in the morning, and I haven't drunk enough coffee, for my brain to work good enough for any more math already...
Title: Re: Isometric Mapping Demo
Post by: STxAxTIC on November 19, 2019, 08:47:55 am
My first thought on this is - what's the end result supposed to look like? That totally influences the beginning...

... Because insert spiel about doing things in proper 3D ... but forgetting that for a second ...

It seems that getting the pseudo-height correct isn't going to be the main problem, but is instead going to be plotting. If you have monsters, projectiles, speech bubbles, or whatever else, the question is how to do draw them in the proper layers so you don't have overlaps? The quick solution I see is to stick your data layers together like a cake, and then slice the scene not horizontally, but in the so-called z-direction (int the screen) from background to foreground. That way, you can just do the painter's algorithm, which is especially easy because of the locked perspective.




Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 19, 2019, 09:09:07 am
You draw it farthest to closest, or, in this case, from 0 to 9.

No need for a spiel on “proper 3D” here; that’s not what we’re doing.  All we’re shooting for here is something like the screenshot below, which is a nice isometric map:

Title: Re: Isometric Mapping Demo
Post by: STxAxTIC on November 19, 2019, 09:27:43 am
If you were just shooting for the screenshot, the project must be finished.

What was the question again?
Title: Re: Isometric Mapping Demo
Post by: Cobalt on November 19, 2019, 10:26:09 am
set your grid size to 10 and you get some issues in the Iso3d view. and the height doesn't seem to be affected by changing the gridsize either.
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 19, 2019, 10:33:31 am
set your grid size to 10 and you get some issues in the Iso3d view. and the height doesn't seem to be affected by changing the gridsize either.

That’s from the current paint fill routine, as I mentioned above.  I’ll work in a different fill method later, which won’t suffer from that problem.  ;)

(Honestly, I think all I really need to do is draw the grid in a separate step and not during the line drawing/painting process.  It’s those stray white lines that act as a boundary, which I think is the issue.  I’ll do some testing later to see if my assumption is correct, or not.). ;)
Title: Re: Isometric Mapping Demo
Post by: Cobalt on November 19, 2019, 11:23:41 am
Ah, Rereading that its clicking now, must have just been not quite awake the first time through(nightmares of Frost Blossom keeping me up! XD ).

Wonder how hard it would be to make a LINE()-(),,BF that could be used for NON-Square shapes made, which would probably be easier with a TRIANGLE()-()-(), LINECOLOR, FILLCOLOR kind of thing, where FILLCOLOR would act like BF in the line statement so if a fill color is present it would fill otherwise it would just be lines. Would take 2 calls to make a "box" but might be easier in the end. maybe manipulating the image memory directly to fill it.
Title: Re: Isometric Mapping Demo
Post by: Petr on November 19, 2019, 01:21:56 pm
Hi Steve.
Thank you for a nice demonstration. I would solve the textures differently. I would use MAPTRIANGLE 2D because it uses the same coordinate system as normal 2D graphics. I have to study your code deeply because I still miss the Z-axis calculation, even though I'm on steps... At the moment, I have come across the ceiling of my abilities.

I do now some experiments with this total pseudo 3D code:

Code: QB64: [Select]
  1. 'Pseudo3D Cube
  2. SCREEN _NEWIMAGE(640, 480, 32)
  3. 'rotation point is 320, 240. Rotation radius is 100.
  4. 'is needed 8* X, 8* Y, 8* Z
  5.  
  6. DIM X(8), Y(8), Z(8)
  7.  
  8.  
  9. texture = _NEWIMAGE(100, 100, 32)
  10. _DEST texture
  11. CLS , &H0DEED0C4
  12. FOR f = 0 TO 100 STEP 20
  13.     LINE (50 + f, 50 + f)-(50 - f, 50 - f), _RGB32(RND * 255, RND * 255, RND * 255), B
  14. r1 = 100
  15. r2 = 50
  16.  
  17.     i$ = INKEY$
  18.     SELECT CASE i$
  19.         CASE " ": fill = fill + 1: IF fill > 1 THEN fill = 0
  20.         CASE "+": r1 = r1 + 1: r2 = r2 + 1
  21.         CASE "-": r1 = r1 - 1: r2 = r2 - 1
  22.         CASE "*": XX1 = XX1 + 1
  23.         CASE "/": YY1 = YY1 + 1
  24.         CASE "9": XX2 = XX2 + 1
  25.         CASE "8": YY2 = YY2 + 1
  26.  
  27.         CASE "7": XX1 = XX1 - 1
  28.         CASE "6": YY1 = YY1 - 1
  29.         CASE "5": XX2 = XX2 - 1
  30.         CASE "4": YY2 = YY2 - 1
  31.  
  32.  
  33.  
  34.     END SELECT
  35.  
  36.     FOR L = 1 TO 4
  37.         angle = angle + _PI / 2
  38.         X(L) = XX1 + 320 + SIN(angle + r) * r1
  39.         Y(L) = YY1 + 170 + COS(angle + r) * r2
  40.  
  41.         X(4 + L) = XX2 + 320 + SIN(angle + r) * r1
  42.         Y(4 + L) = YY2 + 240 + COS(angle + r) * r2
  43.     NEXT
  44.     angle = 0
  45.     CLS
  46.     SELECT CASE fill
  47.         CASE 0
  48.             FOR d = 1 TO 3
  49.                 LINE (X(d), Y(d))-(X(d + 1), Y(d + 1))
  50.                 LINE (X(d + 4), Y(d + 4))-(X(d + 5), Y(d + 5))
  51.                 LINE (X(d), Y(d))-(X(d + 4), Y(d + 4))
  52.             NEXT
  53.             LINE (X(1), Y(1))-(X(4), Y(4))
  54.             LINE (X(5), Y(5))-(X(8), Y(8))
  55.             LINE (X(4), Y(4))-(X(8), Y(8))
  56.  
  57.         CASE 1
  58.             FOR d = 1 TO 3
  59.                 '   EXIT FOR
  60.                 _MAPTRIANGLE (0, 0)-(99, 0)-(0, 99), texture TO(X(d), Y(d))-(X(d + 1), Y(d + 1))-(X(d + 4), Y(d + 4))
  61.                 _MAPTRIANGLE (99, 0)-(0, 99)-(99, 99), texture TO(X(d + 1), Y(d + 1))-(X(d + 4), Y(d + 4))-(X(d + 5), Y(d + 5))
  62.  
  63.  
  64.             NEXT
  65.  
  66.  
  67.             _MAPTRIANGLE (0, 0)-(99, 0)-(0, 99), texture TO(X(1), Y(1))-(X(4), Y(4))-(X(5), Y(5)) 'last side
  68.             _MAPTRIANGLE (99, 0)-(0, 99)-(99, 99), texture TO(X(4), Y(4))-(X(5), Y(5))-(X(8), Y(8)) 'last side
  69.  
  70.             _MAPTRIANGLE (0, 0)-(99, 0)-(0, 99), texture TO(X(1), Y(1))-(X(4), Y(4))-(X(2), Y(2)) 'ceil
  71.             _MAPTRIANGLE (99, 0)-(0, 99)-(99, 99), texture TO(X(4), Y(4))-(X(2), Y(2))-(X(3), Y(3))
  72.  
  73.             _MAPTRIANGLE (0, 0)-(99, 0)-(0, 99), texture TO(X(5), Y(5))-(X(8), Y(8))-(X(6), Y(6)) 'bottom
  74.             _MAPTRIANGLE (99, 0)-(0, 99)-(99, 99), texture TO(X(8), Y(8))-(X(6), Y(6))-(X(7), Y(7))
  75.  
  76.  
  77.     END SELECT
  78.     r = r + .01
  79.     _DISPLAY
  80.     _LIMIT 25
  81.  

(Use space, +,-,*,/,9,8,7,6,5,4)

Of course, this is just an experimental program, nothing miraculous is expected of it.
Title: Re: Isometric Mapping Demo
Post by: bplus on November 19, 2019, 03:30:38 pm
Nice work Petr! That space option is really cool!

I am trying to find your post on your start to real 3D rendering that STxAxTIC made comment to a month or 2 ago?
Title: Re: Isometric Mapping Demo
Post by: Petr on November 19, 2019, 04:21:17 pm
This? It use hardware images + hardware MAPTRIANGLE https://www.qb64.org/forum/index.php?topic=300.msg102623#msg102623
Title: Re: Isometric Mapping Demo
Post by: bplus on November 19, 2019, 08:22:09 pm
Steve could you just stack cube images, back to front, bottom to top?
Title: Re: Isometric Mapping Demo
Post by: Lucky_Strikez on November 19, 2019, 09:43:28 pm
Super cool work guys! Keep it going
Title: Re: Isometric Mapping Demo
Post by: AndyA on November 19, 2019, 11:29:02 pm
@Steve

The easiest way to do isometric cubes that I can think of is to:

1) Draw an isometric cube, paint the three iso-squares facing viewer with desired color.

2) GET the image.

3) PUT the cubes using grid coordinates plus appropriate X/Y offsets.

Using this method allows you to create as many different colored iso-cubes as desired at run time.

The other thing is that using PUT is a whole lot faster than painting each individual cube face.

Just my 2 cents :)

Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 19, 2019, 11:43:39 pm
What I’ll probably end up using is just a bunch of textures for the walls and then _MAPTRIANGLE them to the proper coordinates.  It seems the fastest/easiest way to go.
Title: Re: Isometric Mapping Demo
Post by: bplus on November 20, 2019, 12:48:37 am
Forget PAINT use fquad (fill quadrilateral) which is just a double ftri (fill triangle):
Code: QB64: [Select]
  1.  
  2. 'this text was fetched
  3. SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  4.     DIM a&
  5.     a& = _NEWIMAGE(1, 1, 32)
  6.     _DEST a&
  7.     PSET (0, 0), K
  8.     _DEST 0
  9.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  10.     _FREEIMAGE a& '<<< this is important!
  11.  
  12. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is diagonal of x4, y4
  13. SUB fquad (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  14.     ftri x1, y1, x2, y2, x4, y4, K
  15.     ftri x3, y3, x2, y2, x4, y4, K
  16.  

Here is demo: https://www.qb64.org/forum/index.php?topic=1905.0
Title: Re: Isometric Mapping Demo
Post by: bplus on November 20, 2019, 02:58:49 pm
Problem gridsize of 10 fixed:
Code: QB64: [Select]
  1. SCREEN _NEWIMAGE(1024, 720, 32)
  2.  
  3. DIM SHARED GridSize AS INTEGER
  4. CONST Red = &HFFFFAACC, Green = &HFF00FF00
  5. GridSize = 10 ' < 10 is causing problems with PAINT
  6.  
  7.  
  8. DIM grid(9, 9) AS _BYTE 'create a grid to layout our map
  9.  
  10. FOR i = 0 TO 9 'and let's create that map.  It's just going to be 4 walls and a floor...
  11.     grid(i, 0) = 1 'wall
  12.     grid(i, 9) = 1
  13.     grid(0, i) = 1
  14.     grid(9, i) = 1
  15.  
  16. FOR y = 1 TO 8
  17.     FOR x = 1 TO 8
  18.         grid(x, y) = 2 'floor
  19.     NEXT
  20.  
  21.  
  22. 'Now, let's just draw grids one at a time to compare.
  23. _PRINTSTRING (100, 80), "Normal 2D grid"
  24. FOR y = 0 TO 9
  25.     FOR x = 0 TO 9
  26.         'normal grid first
  27.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  28.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  29.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  30.         LINE (xpos, ypos)-(xpos2, ypos2), Kolor, BF
  31.         LINE (xpos, ypos)-(xpos2, ypos2), &HFFFFFFFF, B
  32.         'and that's all it takes for the normal grid.  Can't get any simpler for 2d maps!
  33.     NEXT
  34.  
  35.  
  36. _PRINTSTRING (350, 80), "Normal 2D grid in Isometic Perspective"
  37. FOR y = 0 TO 9
  38.     FOR x = 0 TO 9
  39.         'And now, let's do the same thing with our isometic map.
  40.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  41.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  42.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  43.         IsoLine xpos, ypos, xpos2, ypos2, 500, 0, Kolor
  44.         'And that's basically all the takes to rotate our map to make it a 2D isometic perspective
  45.     NEXT
  46.  
  47.  
  48. _PRINTSTRING (350, 360), "Normal 2D grid in 3D Isometic Perspective"
  49. FOR y = 0 TO 9
  50.     FOR x = 0 TO 9
  51.         'And here, I'm going to make it a 3D isometic map
  52.         IF grid(x, y) = 1 THEN Kolor = Red ELSE Kolor = Green
  53.         IF grid(x, y) = 1 THEN z = 16 ELSE z = 0 'Give my walls a height of 16, for a cube
  54.  
  55.         xpos = x * GridSize + 100: ypos = y * GridSize + 100
  56.         xpos2 = xpos + GridSize: ypos2 = ypos + GridSize
  57.         IsoLine3D xpos, ypos, xpos2, ypos2, z, 500, 300, Kolor
  58.     NEXT
  59.  
  60.  
  61.  
  62.  
  63.  
  64.  
  65.  
  66.  
  67.  
  68. FUNCTION CX2I (x AS LONG, y AS LONG) 'Convert Cartesian X To Isometic coordinates
  69.     CX2I = x - y
  70.  
  71. FUNCTION CY2I (x AS LONG, y AS LONG) 'Convert Cartesian Y To Isometic coordinates
  72.     CY2I = (x + y) / 2
  73.  
  74. SUB IsoLine (x, y, x2, y2, xoffset, yoffset, kolor AS _UNSIGNED LONG)
  75.     'since we're drawing a diamond and not a square box, we can't use Line BF.
  76.     'We have to manually down the 4 points of the line.
  77.     LINE (CX2I(x, y) + xoffset, CY2I(x, y) + yoffset)-(CX2I(x2, y) + xoffset, CY2I(x2, y) + yoffset), kolor
  78.     LINE -(CX2I(x2, y2) + xoffset, CY2I(x2, y2) + yoffset), kolor
  79.     LINE -(CX2I(x, y2) + xoffset, CY2I(x, y2) + yoffset), kolor
  80.     LINE -(CX2I(x, y) + xoffset, CY2I(x, y) + yoffset), kolor
  81.     PAINT (CX2I(x, y) + xoffset, CY2I(x, y) + 4), kolor 'and fill the diamond solid
  82.     LINE (CX2I(x, y) + xoffset, CY2I(x, y) + yoffset)-(CX2I(x2, y) + xoffset, CY2I(x2, y) + yoffset), &HFFFFFFFF
  83.     LINE -(CX2I(x2, y2) + xoffset, CY2I(x2, y2) + yoffset), &HFFFFFFFF
  84.     LINE -(CX2I(x, y2) + xoffset, CY2I(x, y2) + yoffset), &HFFFFFFFF
  85.     LINE -(CX2I(x, y) + xoffset, CY2I(x, y) + yoffset), &HFFFFFFFF
  86.  
  87. SUB IsoLine3D (x, y, x2, y2, z, xoffset, yoffset, kolor AS _UNSIGNED LONG)
  88.     'Like IsoLine, we're going to have to draw our lines manually.
  89.     'only in this case, we also need a Z coordinate to tell us how THICK/TALL/HIGH to make our tile
  90.     r = _RED32(kolor): g = _GREEN32(kolor): b = _BLUE32(kolor)
  91.     'Let's just do all the math first this time.
  92.     'We need to turn those 4 normal points into 4 isometric points (x, y, x1, y1)
  93.  
  94.     TempX1 = CX2I(x, y) + xoffset: TempY1 = CY2I(x, y) + yoffset
  95.     TempX2 = CX2I(x2, y) + xoffset: TempY2 = CY2I(x2, y) + yoffset
  96.     TempX3 = CX2I(x2, y2) + xoffset: TempY3 = CY2I(x2, y2) + yoffset
  97.     TempX4 = CX2I(x, y2) + xoffset: TempY4 = CY2I(x, y2) + yoffset
  98.  
  99.     'The top
  100.     'LINE (TempX1, TempY1 - z)-(TempX2, TempY2 - z), kolor 'draw the diamond
  101.     'LINE -(TempX3, TempY3 - z), kolor
  102.     'LINE -(TempX4, TempY4 - z), kolor
  103.     'LINE -(TempX1, TempY1 - z), kolor
  104.     'PAINT (TempX1, TempY1 - z + 4), kolor 'and fill the diamond solid
  105.     fquad TempX1, TempY1 - z, TempX2, TempY2 - z, TempX3, TempY3 - z, TempX4, TempY4 - z, kolor
  106.     '
  107.     'LINE (TempX1, TempY1 - z)-(TempX2, TempY2 - z), -1 'and redraw the grid
  108.     'LINE -(TempX3, TempY3 - z), -1
  109.     'LINE -(TempX4, TempY4 - z), -1
  110.     'LINE -(TempX1, TempY1 - z), -1
  111.  
  112.     IF z <> 0 THEN 'no need to draw any height, if there isn't any.
  113.         'the left side
  114.         'LINE (TempX4, TempY4 - z)-(TempX4, TempY4), kolor 'draw it
  115.         'LINE -(TempX3, TempY3), kolor
  116.         'LINE -(TempX3, TempY3 - z), kolor
  117.         'LINE -(TempX4, TempY4 - z), kolor
  118.         'PAINT (TempX3 - 2, TempY3 - z + 2), kolor 'fill it
  119.         fquad TempX4, TempY4 - z, TempX4, TempY4, TempX3, TempY3, TempX3, TempY3 - z, _RGB32(.25 * r, .5 * g, .75 * b)
  120.         '
  121.         'LINE (TempX4, TempY4 - z)-(TempX4, TempY4), -1 'redraw the grid lines
  122.         'LINE -(TempX3, TempY3), -1
  123.         'LINE -(TempX3, TempY3 - z), -1
  124.         'LINE -(TempX4, TempY4 - z), -1
  125.         'and then for the right side
  126.         'LINE (TempX3, TempY3 - z)-(TempX3, TempY3), kolor 'draw it
  127.         'LINE -(TempX2, TempY2), kolor
  128.         'LINE -(TempX2, TempY2 - z), kolor
  129.         'LINE -(TempX3, TempY3 - z), kolor
  130.         'PAINT (TempX3 + 2, TempY3 - z + 2), kolor 'fill it
  131.         fquad TempX3, TempY3 - z, TempX3, TempY3, TempX2, TempY2, TempX2, TempY2 - z, _RGB32(.75 * r, .3 * g, .3 * b)
  132.  
  133.         'LINE (TempX3, TempY3 - z)-(TempX3, TempY3), -1 'redraw the grid lines
  134.         'LINE -(TempX2, TempY2), -1
  135.         'LINE -(TempX2, TempY2 - z), -1
  136.         'LINE -(TempX3, TempY3 - z), -1
  137.     END IF
  138.  
  139.  
  140.  
  141.  
  142. 'this text was fetched
  143. SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  144.     DIM a&
  145.     a& = _NEWIMAGE(1, 1, 32)
  146.     _DEST a&
  147.     PSET (0, 0), K
  148.     _DEST 0
  149.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  150.     _FREEIMAGE a& '<<< this is important!
  151.  
  152. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is diagonal of x4, y4
  153. SUB fquad (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  154.     ftri x1, y1, x2, y2, x4, y4, K
  155.     ftri x3, y3, x2, y2, x4, y4, K
  156.  
  157.  
  158.  

I also removed all the line drawing for edges and made slight change to red color
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 20, 2019, 03:14:25 pm
Code: [Select]
SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
    STATIC a&
    D = _DEST
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
     _DEST a&
    PSET (0, 0), K
    _DEST D
    _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
END SUB

A few quick changes to the routine which should speed it up just a wee bit.  Speed is one of the primary concerns with any sort of fill routine.  ;)

(Also returned the _DEST to whatever it was previously, and not just the display screen by default.)
Title: Re: Isometric Mapping Demo
Post by: bplus on November 20, 2019, 03:23:38 pm
Code: [Select]
SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
    STATIC a&
    D = _DEST
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
     _DEST a&
    PSET (0, 0), K
    _DEST D
    _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
END SUB

A few quick changes to the routine which should speed it up just a wee bit.  Speed is one of the primary concerns with any sort of fill routine.  ;)

(Also returned the _DEST to whatever it was previously, and not just the display screen by default.)

With STATIC a&, you don't have to _FREEIMAGE a& ?
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 20, 2019, 03:47:25 pm
With STATIC a&, you don't have to _FREEIMAGE a& ?

Nope.  You just make it once, leave it always available, and forget about it.  At its size, what is it?  4 bytes in memory + whatever the internal header structure uses?  Why reserve it, use it, free it, over and over, at that size?
Title: Re: Isometric Mapping Demo
Post by: _vince on November 20, 2019, 04:35:02 pm
Looking good, Steve!
Title: Re: Isometric Mapping Demo
Post by: Cobalt on November 21, 2019, 12:27:49 pm
Would it be possible to make a routine that would make each individual iso-box 1 at a time. like SUB ISOBOX(X,Y,SIZE,HEIGHT) and then you just have to call that over and over to create a grid. No that I think about that might as well name it CUBE.

would It be able to processes the cubes in stages? draw the outer shell, then fill then draw the inner lines? I added an Image of what I mean, as I am terrible at explaining!

thought if it was handled this way using paint might still be an easier and viable solution. might beable to add an option as to if the "back side" lines are drawn or not. (so the cube looks solid not semi transparent)

Title: Re: Isometric Mapping Demo
Post by: SMcNeill on November 21, 2019, 12:32:14 pm
Would it be possible to make a routine that would make each individual iso-box 1 at a time. like SUB ISOBOX(X,Y,SIZE,HEIGHT) and then you just have to call that over and over to create a grid. No that I think about that might as well name it CUBE.

would It be able to processes the cubes in stages? draw the outer shell, then fill then draw the inner lines? I added an Image of what I mean, as I am terrible at explaining!

thought if it was handled this way using paint might still be an easier and viable solution. might beable to add an option as to if the "back side" lines are drawn or not. (so the cube looks solid not semi transparent)

You can, as long as the colors are all the same on each side, but what do you do if they’re different?  (Brown dirt sides for a cliff, green grass on top, as an example.)
Title: Re: Isometric Mapping Demo
Post by: bplus on November 21, 2019, 12:39:57 pm
Would it be possible to make a routine that would make each individual iso-box 1 at a time. like SUB ISOBOX(X,Y,SIZE,HEIGHT) and then you just have to call that over and over to create a grid. No that I think about that might as well name it CUBE.

would It be able to processes the cubes in stages? draw the outer shell, then fill then draw the inner lines? I added an Image of what I mean, as I am terrible at explaining!

thought if it was handled this way using paint might still be an easier and viable solution. might beable to add an option as to if the "back side" lines are drawn or not. (so the cube looks solid not semi transparent)

I did just that with this: https://www.qb64.org/forum/index.php?topic=1905.0 and I even named it cube2 as cube was from other code I used for model.

There all sorts of cubes you can make. The one demo'd above fit in convenient square for front face 1 quarter of square for easy overlapping of psuedo 3D model, you can put the square face in any of 4 corners, or you can do regular hexagon shapes and grid them out like Hexagonal Life or Hexagonal Minesweeper.

You can use MAPTRIANGLE (2D) to add texture fills instead of ftri or fquad, which use MAPTRIANGLE too, and has to be much faster and more effective than PAINT eg transparent color fills!
Title: Re: Isometric Mapping Demo
Post by: Cobalt on November 21, 2019, 07:07:12 pm
You can, as long as the colors are all the same on each side, but what do you do if they’re different?  (Brown dirt sides for a cliff, green grass on top, as an example.)

what about using iso-boxes? have a horizontal one for top and bottom and then vertical ones for the sides? then each side could be a different color, or even alpha shaded? granted you would have 6 calls for each cube. I don't know, just fartin ideas that could keep it simple line\paint based routines.

You can use MAPTRIANGLE (2D) to add texture fills instead of ftri or fquad, which use MAPTRIANGLE too, and has to be much faster and more effective than PAINT eg transparent color fills!

I mean Maptri would be great for applying textures to the faces, but at sizes <32(or maybe even 24-pushing though) a texture is going to be very limited. I've used DirectQB's maptri function to texture walls in a '3D'ish type dungeon crawl, but I was using full screen 13, textured tris up to 256px hor, and 192px vert. my source texture was 256x256 even for that.(had to be ^2 for DQB though so some of that was lost[if i remember correctly, was over 20y/a now.])
I should try to port that to QB64 actually. Use higher res textures for 640x400 or 800x600.
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 11:12:53 am
Code: [Select]
SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
    STATIC a&
    D = _DEST
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
     _DEST a&
    PSET (0, 0), K
    _DEST D
    _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
END SUB

A few quick changes to the routine which should speed it up just a wee bit.  Speed is one of the primary concerns with any sort of fill routine.  ;)

(Also returned the _DEST to whatever it was previously, and not just the display screen by default.)

Hi Steve,

I found problem with this modified routine of ftri plus a place where neither version of cSleep or your Pause worked.
Code: QB64: [Select]
  1. _TITLE "fTri Tests, click 4 points to draw transparent quad" 'b+ 2019-12-15 discovered a problem with fTri or using it with fQuad
  2.  
  3. CONST xmax = 800, ymax = 600
  4. SCREEN _NEWIMAGE(xmax, ymax, 32)
  5. _SCREENMOVE 300, 40
  6. DIM pi, oldMouse
  7. DIM mx(1 TO 4), my(1 TO 4)
  8.     CLS
  9.     WHILE pi < 4 'get 4 mouse clicks
  10.         _PRINTSTRING (5, 5), SPACE$(20)
  11.         _PRINTSTRING (5, 5), "Need 4 clicks, have" + STR$(pi)
  12.         WHILE _MOUSEINPUT: WEND
  13.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN 'new mouse down
  14.             pi = pi + 1
  15.             mx(pi) = _MOUSEX: my(pi) = _MOUSEY
  16.             CIRCLE (mx(pi), my(pi)), 2
  17.         END IF
  18.         oldMouse = _MOUSEBUTTON(1)
  19.         _DISPLAY
  20.         _LIMIT 60
  21.     WEND
  22.     _PRINTSTRING (5, 5), "Sleeping, click or press key... or esc"
  23.     fquad1 mx(1), my(1), mx(2), my(2), mx(3), my(3), mx(4), my(4), &H880000FF
  24.     _DISPLAY
  25.     ' cSleep 45   '<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< dang!! this is working either !!!!!!!!!!!!!!!!!!!!
  26.     ' Pause 30 ' <<<<<<<<<<<<<<<<<<<< well this isn't working either!!!!!!!!!!!!!!
  27.     SLEEP
  28.     pi = 0 'point index
  29.  
  30.  
  31. ' my original fTri that never had a problem with
  32. ' found at [abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]:    http://www.[abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]/forum/index.php?topic=14425.0
  33. SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  34.     DIM D AS LONG, a&
  35.     D = _DEST
  36.     a& = _NEWIMAGE(1, 1, 32)
  37.     _DEST a&
  38.     PSET (0, 0), K
  39.     _DEST D
  40.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  41.     _FREEIMAGE a& '<<< this is important!
  42.  
  43. '2019-11-20 Steve saves some time with STATIC and saves and restores last dest
  44. SUB ftri1 (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  45.     DIM D AS LONG
  46.     STATIC a&
  47.     D = _DEST
  48.     IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
  49.     _DEST a&
  50.     PSET (0, 0), K
  51.     _DEST D
  52.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  53.  
  54. ' original fill quad that may be at fault using Steve's fTri version
  55. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is opposite of x4, y4
  56. SUB fquad (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  57.     ftri1 x1, y1, x2, y2, x4, y4, K
  58.     ftri1 x3, y3, x2, y2, x4, y4, K
  59.  
  60. 'try a new quad   OK works fine with original fTri but not fTr1 Steves mod
  61. SUB fquad1 (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  62.     ftri1 x1, y1, x2, y2, x3, y3, K
  63.     ftri1 x3, y3, x4, y4, x1, y1, K
  64.  
  65. 'now this original code is not working!
  66. SUB cSleep (secsWait AS DOUBLE) 'wait for keypress or mouseclick, solves midnight problem nicely I think
  67.     DIM wayt AS INTEGER, oldMouse AS INTEGER, k AS LONG, startTime AS DOUBLE
  68.  
  69.     startTime = TIMER
  70.     wayt = 1
  71.     _KEYCLEAR
  72.     WHILE wayt
  73.         WHILE _MOUSEINPUT: WEND
  74.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN wayt = 0
  75.         oldMouse = _MOUSEBUTTON(1) ' <<< this is Steve's cool way to get clear of mouse click
  76.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: wayt = 0
  77.         IF TIMER - startTime < 0 THEN 'past midnight
  78.             IF TIMER + 24 * 60 * 60 - startTime > secsWait THEN wayt = 0
  79.         ELSE
  80.             IF TIMER - startTime >= secsWait THEN wayt = 0
  81.         END IF
  82.         _LIMIT 30
  83.     WEND
  84.  
  85. 'try Steve's Pause, nope doesn't work either!!!!
  86. SUB Pause (time AS _FLOAT)
  87.     DIM ExitTime AS _FLOAT, oldMouse AS INTEGER, k AS LONG
  88.     _KEYCLEAR 'clear the keyboard buffer so we don't automatically exit the routine
  89.     IF time <= 0 THEN ExitTime = 1.18E+1000 ELSE ExitTime = time + TIMER
  90.     DO
  91.         WHILE _MOUSEINPUT: WEND: IF _MOUSEBUTTON(1) AND NOT oldMouse THEN EXIT SUB
  92.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: EXIT SUB 'clear any stray key events so they don't mess with code outside the Pause.
  93.         oldMouse = _MOUSEBUTTON(1)
  94.         _LIMIT 10
  95.     LOOP UNTIL ExitTime < TIMER
  96.  
  97.  
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 11:50:45 am
Hi Steve,

This line added to both cSleep and Pause seems to fix the problem in the code above:
Code: QB64: [Select]
where we use _KEYCLEAR, we have to clear old mouse clicks also?

I have not tested updated code elsewhere yet.
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on December 16, 2019, 12:54:35 pm
Hi Steve,

This line added to both cSleep and Pause seems to fix the problem in the code above:
Code: QB64: [Select]
where we use _KEYCLEAR, we have to clear old mouse clicks also?

I have not tested updated code elsewhere yet.

One thing I see is missing: oldmouse should be set to -1 before the loop starts.

WHILE _MOUSEINPUT: WEND
IF _MOUSEBUTTON(1) = -1 AND oldmouse = 0 THEN...

If we don’t set oldmouse to -1, we’d instantly kick out of the loop if the user had the button down to begin with.

By setting oldmouse to -1 at the start, we treat the routine as if the button was previously held down, and only accept a click once we get confirmation it’s been up, before being pressed down.

^There might be your error.
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 01:25:39 pm
Hi Steve,

That Pause routine came from here:
https://www.qb64.org/forum/index.php?topic=1511.msg111606#msg111606
and was what I based cSleep on.


Setting oldMouse to -1 at start does seem better than my fix. I will test, thanks :)

Any thoughts about fTri? It seems to retain the last transparent color when we use STATIC a& and not _FREEIMAGE.
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on December 16, 2019, 01:32:35 pm
Hi Steve,

That Pause routine came from here:
https://www.qb64.org/forum/index.php?topic=1511.msg111606#msg111606
and was what I based cSleep on.


Setting oldMouse to -1 at start does seem better than my fix. I will test, thanks :)

Any thoughts about fTri? It seems to retain the last transparent color when we use STATIC a& and not _FREEIMAGE.

Several options; as I didn’t think about alpha blending with the static handle.

CLS, 0 could clear the old pixel first.
Or _DONTBLEND a& before using the PSET command (I’d probably just go this route, as DONTBLEND is a speed up over using blend.)

Edit: Went in and preset oldmouse to -1 in the Pause command so it shouldn’t have the previously down glitch anymore.  ;)

Code: [Select]
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32): _DONTBLEND a&
(The above should set a& to no blend mode and then leave it to that, fixing the alpha issue with it.)
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 02:16:04 pm
Several options; as I didn’t think about alpha blending with the static handle.

CLS, 0 could clear the old pixel first.
Or _DONTBLEND a& before using the PSET command (I’d probably just go this route, as DONTBLEND is a speed up over using blend.)

Edit: Went in and preset oldmouse to -1 in the Pause command so it shouldn’t have the previously down glitch anymore.  ;)

Code: [Select]
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32): _DONTBLEND a&
(The above should set a& to no blend mode and then leave it to that, fixing the alpha issue with it.)

Hi Steve,

Reading about _DONTBLEND in help, it says that such will make the background opaque and block the background. When I use transparent colors with fTri, I usually want just a layer of see through color over what's behind. It's looking like I want to stick with original fTri that frees image to keep pure transparent layers going.

Update: confirmed the transparency is lost with _DONTBLEND

Thanks for your thoughts :)
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on December 16, 2019, 02:32:53 pm
Hi Steve,

Reading about _DONTBLEND in help, it says that such will make the background opaque and block the background. When I use transparent colors with fTri, I usually want just a layer of see through color over what's behind. It's looking like I want to stick with original fTri that frees image to keep pure transparent layers going.

Update: confirmed the transparency is lost with _DONTBLEND

Thanks for your thoughts :)

I'm lost as to what the heck is wrong with _DONTBLEND; it appears to work as it should for me:

Code: QB64: [Select]
  1. SCREEN _NEWIMAGE(640, 480, 32)
  2. PSET (0, 0), &HAAFFFFFF

How is the transparency lost?  When _dontblend is used, all we do is set the color to exactly what we specify it to be. 

As for the wiki, I have no idea what the heck this is supposed to mean:

Quote
Use CLS or _DONTBLEND to make a new surface background _ALPHA 255 or opaque.

_DONTBLEND doesn't change the alpha levels.  Unless my experience using it is just overly weird, all it does for us is directly put our specified colors to the image, without trying to blend the alpha levels and such like normal. 

Is there some secret trick to this command that I've been missing all these years??
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 03:08:20 pm
Hi Steve,

Here is my test code for the _DONTBLEND. First run it to see the shade of transparent used with fTri that uses _FREEIMAGE. Then run main code with fQuad1 that runs fTri with the _DONTBLEND, you can see it is full opaque blue and so much lighter because solid.
Code: QB64: [Select]
  1. _TITLE "fTri Tests, click 4 points to draw transparent quad" 'b+ 2019-12-15 discovered a problem with fTri or using it with fQuad
  2.  
  3. CONST xmax = 800, ymax = 600
  4. SCREEN _NEWIMAGE(xmax, ymax, 32)
  5. _SCREENMOVE 300, 40
  6. DIM pi, oldMouse
  7. DIM mx(1 TO 4), my(1 TO 4)
  8.     CLS
  9.     oldMouse = -1
  10.     WHILE pi < 4 'get 4 mouse clicks
  11.         _PRINTSTRING (5, 5), SPACE$(20)
  12.         _PRINTSTRING (5, 5), "Need 4 clicks, have" + STR$(pi)
  13.         WHILE _MOUSEINPUT: WEND
  14.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN 'new mouse down
  15.             pi = pi + 1
  16.             mx(pi) = _MOUSEX: my(pi) = _MOUSEY
  17.             CIRCLE (mx(pi), my(pi)), 2
  18.         END IF
  19.         oldMouse = _MOUSEBUTTON(1)
  20.         _DISPLAY
  21.         _LIMIT 60
  22.     WEND
  23.     _PRINTSTRING (5, 5), "Sleeping, click or press key... or esc"
  24.     'fquad1 mx(1), my(1), mx(2), my(2), mx(3), my(3), mx(4), my(4), &H880000FF '>>>>>>>>>>>>>>>>>>> switch between fquad and fquad1
  25.     fquad mx(1), my(1), mx(2), my(2), mx(3), my(3), mx(4), my(4), &H880000FF '>>>>>>>>>>>>>>>>>>> switch between fquad and fquad1
  26.     _DISPLAY
  27.     cSleep 45 '<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< dang!! this is working either fixed 2019-12-16 2nd time oldMouse = -1 at start
  28.     'Pause 30 ' <<<<<<<<<<<<<<<<<<<< well this isn't working either!!!!!!!!!!!!!! now fixed (2x's, Steves fix was better)
  29.     'SLEEP
  30.     pi = 0 'point index
  31.  
  32.  
  33. ' my original fTri that never had a problem with
  34. ' found at [abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]:    http://www.[abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]/forum/index.php?topic=14425.0
  35. SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  36.     DIM D AS LONG, a&
  37.     D = _DEST
  38.     a& = _NEWIMAGE(1, 1, 32)
  39.     _DEST a&
  40.     PSET (0, 0), K
  41.     _DEST D
  42.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  43.     _FREEIMAGE a& '<<< this is important!
  44.  
  45. '2019-11-20 Steve saves some time with STATIC and saves and restores last dest
  46. SUB ftri1 (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  47.     DIM D AS LONG
  48.     STATIC a&
  49.     D = _DEST
  50.     IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32): _DONTBLEND a& '<< fix ??
  51.     _DEST a&
  52.     PSET (0, 0), K
  53.     _DEST D
  54.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  55.  
  56. ' original fill quad that may be at fault using Steve's fTri version
  57. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is opposite of x4, y4
  58. SUB fquad1 (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  59.     ftri1 x1, y1, x2, y2, x4, y4, K
  60.     ftri1 x3, y3, x2, y2, x4, y4, K
  61.  
  62. 'update 2019-12-16 needs orig fTri
  63. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is opposite of x4, y4
  64. SUB fquad (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  65.     ftri x1, y1, x2, y2, x3, y3, K
  66.     ftri x3, y3, x4, y4, x1, y1, K
  67.  
  68. 'update 2019-12-16 in ...test\graphics\fTri tests.bas
  69. SUB cSleep (secsWait AS DOUBLE) 'wait for keypress or mouseclick, solves midnight problem nicely I think
  70.     DIM wayt AS INTEGER, oldMouse AS INTEGER, k AS LONG, startTime AS DOUBLE
  71.  
  72.     oldMouse = -1 '2019-12-16 2nd fix today assume an old mouse click is still active
  73.     startTime = TIMER
  74.     wayt = 1
  75.     _KEYCLEAR
  76.     WHILE wayt
  77.         WHILE _MOUSEINPUT: WEND
  78.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN wayt = 0
  79.         oldMouse = _MOUSEBUTTON(1) ' <<< this is Steve's cool way to get clear of mouse click
  80.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: wayt = 0
  81.         IF TIMER - startTime < 0 THEN 'past midnight
  82.             IF TIMER + 24 * 60 * 60 - startTime > secsWait THEN wayt = 0
  83.         ELSE
  84.             IF TIMER - startTime >= secsWait THEN wayt = 0
  85.         END IF
  86.         _LIMIT 30
  87.     WEND
  88.  
  89. 'try Steve's Pause, nope doesn't work either!!!!
  90. SUB Pause (time AS _FLOAT)
  91.     DIM ExitTime AS _FLOAT, oldMouse AS INTEGER, k AS LONG
  92.  
  93.     _KEYCLEAR 'clear the keyboard buffer so we don't automatically exit the routine
  94.     'WHILE _MOUSEINPUT OR _MOUSEBUTTON(1): WEND 'clear mouse too
  95.     oldMouse = -1 'in case a mouse down still active
  96.     IF time <= 0 THEN ExitTime = 1.18E+1000 ELSE ExitTime = time + TIMER
  97.     DO
  98.         WHILE _MOUSEINPUT: WEND: IF _MOUSEBUTTON(1) AND NOT oldMouse THEN EXIT SUB
  99.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: EXIT SUB 'clear any stray key events so they don't mess with code outside the Pause.
  100.         oldMouse = _MOUSEBUTTON(1)
  101.         _LIMIT 10
  102.     LOOP UNTIL ExitTime < TIMER
  103.  
  104.  
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on December 16, 2019, 03:09:04 pm
Doing some testing at the computer, I sorted out what was wrong with what I told you:

Code: [Select]
SUB ftri1 (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
    DIM D AS LONG
    STATIC a&
    D = _DEST
    IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
    _DEST a&
    _DONTBLEND a& 'set the image to not blend, so your new color will set itself without blending
    PSET (0, 0), K
    _BLEND a& 'but then turn blending on for the image, so when you use _MAPTRIANGLE, it'll blend into the other screen as desired.
    _DEST D
    _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
END SUB

See if the above doesn't work as intended.  ;)

We can't leave blending off with an image, if we want to blend it with another image.  (Which is kind of oddish.  I would think that all we'd care about is the _DEST blending state, not the _SOURCE's as well...)

Have blending off when setting your pixel, then turn it back on before making use of the _MAPTRIANGLE command.  It should fix the issue.  (I think.)


EDIT:

Testing, I think the code works as intended now:
Code: QB64: [Select]
  1. _TITLE "fTri Tests, click 4 points to draw transparent quad" 'b+ 2019-12-15 discovered a problem with fTri or using it with fQuad
  2.  
  3. CONST xmax = 800, ymax = 600
  4. SCREEN _NEWIMAGE(xmax, ymax, 32)
  5. _SCREENMOVE 300, 40
  6. DIM pi, oldMouse
  7. DIM mx(1 TO 4), my(1 TO 4)
  8.     CLS
  9.     oldMouse = -1
  10.     WHILE pi < 4 'get 4 mouse clicks
  11.         _PRINTSTRING (5, 5), SPACE$(20)
  12.         _PRINTSTRING (5, 5), "Need 4 clicks, have" + STR$(pi)
  13.         WHILE _MOUSEINPUT: WEND
  14.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN 'new mouse down
  15.             pi = pi + 1
  16.             mx(pi) = _MOUSEX: my(pi) = _MOUSEY
  17.             CIRCLE (mx(pi), my(pi)), 2
  18.         END IF
  19.         oldMouse = _MOUSEBUTTON(1)
  20.         _DISPLAY
  21.         _LIMIT 60
  22.     WEND
  23.     _PRINTSTRING (5, 5), "Sleeping, click or press key... or esc"
  24.     'fquad1 mx(1), my(1), mx(2), my(2), mx(3), my(3), mx(4), my(4), &H880000FF '>>>>>>>>>>>>>>>>>>> switch between fquad and fquad1
  25.     fquad mx(1), my(1), mx(2), my(2), mx(3), my(3), mx(4), my(4), &H880000FF '>>>>>>>>>>>>>>>>>>> switch between fquad and fquad1
  26.     _DISPLAY
  27.     cSleep 45 '<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<< dang!! this is working either fixed 2019-12-16 2nd time oldMouse = -1 at start
  28.     'Pause 30 ' <<<<<<<<<<<<<<<<<<<< well this isn't working either!!!!!!!!!!!!!!
  29.     'SLEEP
  30.     pi = 0 'point index
  31.  
  32.  
  33. ' my original fTri that never had a problem with
  34. ' found at [abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]:    http://www.[abandoned, outdated and now likely malicious qb64 dot net website - don’t go there]/forum/index.php?topic=14425.0
  35. SUB ftri (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  36.     DIM D AS LONG, a&
  37.     D = _DEST
  38.     a& = _NEWIMAGE(1, 1, 32)
  39.     _DEST a&
  40.     PSET (0, 0), K
  41.     _DEST D
  42.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  43.     _FREEIMAGE a& '<<< this is important!
  44.  
  45. '2019-11-20 Steve saves some time with STATIC and saves and restores last dest
  46. SUB ftri1 (x1, y1, x2, y2, x3, y3, K AS _UNSIGNED LONG)
  47.     DIM D AS LONG
  48.     STATIC a&
  49.     D = _DEST
  50.     IF a& = 0 THEN a& = _NEWIMAGE(1, 1, 32)
  51.     _DEST a&
  52.     _DONTBLEND a&
  53.     PSET (0, 0), K
  54.     _BLEND a&
  55.     _DEST D
  56.     _MAPTRIANGLE _SEAMLESS(0, 0)-(0, 0)-(0, 0), a& TO(x1, y1)-(x2, y2)-(x3, y3)
  57.  
  58. ' original fill quad that may be at fault using Steve's fTri version
  59. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is opposite of x4, y4
  60. SUB fquad1 (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  61.     ftri1 x1, y1, x2, y2, x4, y4, K
  62.     ftri1 x3, y3, x2, y2, x4, y4, K
  63.  
  64. 'update 2019-12-16 needs orig fTri
  65. 'need 4 non linear points (not all on 1 line) list them clockwise so x2, y2 is opposite of x4, y4
  66. SUB fquad (x1, y1, x2, y2, x3, y3, x4, y4, K AS _UNSIGNED LONG)
  67.     ftri x1, y1, x2, y2, x3, y3, K
  68.     ftri x3, y3, x4, y4, x1, y1, K
  69.  
  70. 'update 2019-12-16 in ...test\graphics\fTri tests.bas
  71. SUB cSleep (secsWait AS DOUBLE) 'wait for keypress or mouseclick, solves midnight problem nicely I think
  72.     DIM wayt AS INTEGER, oldMouse AS INTEGER, k AS LONG, startTime AS DOUBLE
  73.  
  74.     oldMouse = -1 '2019-12-16 2nd fix today assume an old mouse click is still active
  75.     startTime = TIMER
  76.     wayt = 1
  77.     _KEYCLEAR
  78.     WHILE wayt
  79.         WHILE _MOUSEINPUT: WEND
  80.         IF _MOUSEBUTTON(1) AND oldMouse = 0 THEN wayt = 0
  81.         oldMouse = _MOUSEBUTTON(1) ' <<< this is Steve's cool way to get clear of mouse click
  82.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: wayt = 0
  83.         IF TIMER - startTime < 0 THEN 'past midnight
  84.             IF TIMER + 24 * 60 * 60 - startTime > secsWait THEN wayt = 0
  85.         ELSE
  86.             IF TIMER - startTime >= secsWait THEN wayt = 0
  87.         END IF
  88.         _LIMIT 30
  89.     WEND
  90.  
  91. 'try Steve's Pause, nope doesn't work either!!!!
  92. SUB Pause (time AS _FLOAT)
  93.     DIM ExitTime AS _FLOAT, oldMouse AS INTEGER, k AS LONG
  94.  
  95.     _KEYCLEAR 'clear the keyboard buffer so we don't automatically exit the routine
  96.     'WHILE _MOUSEINPUT OR _MOUSEBUTTON(1): WEND 'clear mouse too
  97.     oldMouse = -1 'in case a mouse down still active
  98.     IF time <= 0 THEN ExitTime = 1.18E+1000 ELSE ExitTime = time + TIMER
  99.     DO
  100.         WHILE _MOUSEINPUT: WEND: IF _MOUSEBUTTON(1) AND NOT oldMouse THEN EXIT SUB
  101.         k = _KEYHIT: IF k > 0 THEN _KEYCLEAR: EXIT SUB 'clear any stray key events so they don't mess with code outside the Pause.
  102.         oldMouse = _MOUSEBUTTON(1)
  103.         _LIMIT 10
  104.     LOOP UNTIL ExitTime < TIMER
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 03:26:12 pm
Steve, I think you still have fQuad1 line still commented out and are using fQuad line with the other fTri used.
 I am still seeing a bright solid blue when I uncomment fquad1 and comment fquad.

fquad1 tests with the STATIC a& with _DONTBLEND and fquad tests the working fTri that uses _FREEIMAGE.


Update: OK so I tried testing some other ranges of Alpha and discovered Steve's fix for fTri is working fine and the original fTri isn't registering anything see-able in low ranges of Alpha.

So I am confused, maybe fTri was broken all along? This is sort of the same problem with the version of Titled Ellipse Fill I came up with yesterday, low alpha's are not see-able.
Title: Re: Isometric Mapping Demo
Post by: SMcNeill on December 16, 2019, 10:40:00 pm
Steve, I think you still have fQuad1 line still commented out and are using fQuad line with the other fTri used.
 I am still seeing a bright solid blue when I uncomment fquad1 and comment fquad.

fquad1 tests with the STATIC a& with _DONTBLEND and fquad tests the working fTri that uses _FREEIMAGE.


Update: OK so I tried testing some other ranges of Alpha and discovered Steve's fix for fTri is working fine and the original fTri isn't registering anything see-able in low ranges of Alpha.

So I am confused, maybe fTri was broken all along? This is sort of the same problem with the version of Titled Ellipse Fill I came up with yesterday, low alpha's are not see-able.

Is it just a case of low alpha getting blended out into the background?  Take an image with 255 alpha, blend it with 10 alpha, and it fades it into the background.  Take a background with 0 alpha, blend it with 10 alpha, and it’s going to end up so faded as to be transparent in almost all cases.  (Particularly with my poor eyes an high gamma/contrast settings...)
Title: Re: Isometric Mapping Demo
Post by: bplus on December 16, 2019, 11:06:01 pm
Steve it was the _DONTBLEND / _BLEND thing, not only did it make the Fill Triangle code work better it fixed the problem I had doing tiltled ellipse fills from a Flat Filled Ellipse and RotoZooming the tilt onto the drawing. I guess I misunderstood how drawing transparencies were taking place on a 0 alpha background.