Oxygen Basic

Programming => Example Code => User Interface => Topic started by: Arnold on November 21, 2016, 08:48:48 AM

Title: Japi Experiments
Post by: Arnold on November 21, 2016, 08:48:48 AM
Hello,

while working a little bit with the Japi library last week I found this Julia demo in the OxygenBasic forum:

http://www.oxygenbasic.org/forum/index.php?topic=1144.msg9846#msg9846 (http://www.oxygenbasic.org/forum/index.php?topic=1144.msg9846#msg9846)

The demo is coded in Freebasic and I ported it to OxygenBasic and added some options.

The program should run anywhere on harddisk or usb-stick if Java is installed. It expects Japi in /OxygenBasic/projectsC. If using a previous distribution of OxygenBasic then projectsC must be changed to ProjectsB.

I am not really interested in the Julia algorithm but maybe I could get some help about some of the parameters of the algorithm used in the program and what they are used for?

Roland

Edit: there was a small bug in the save options routine. In the meantime I have modified the correspondent lines.

Code: OxygenBasic
  1. 'Using the Japi library
  2. 'www.japi.de
  3.  
  4. $ filename "julia.exe"
  5. 'include "$/inc/RTL32.inc"
  6. 'include "$/inc/console.inc"
  7.  
  8. extern lib "$/projectsC/Japi/japi.dll" cdecl
  9. include once "$/projectsC/Japi/japi.h"
  10. end extern
  11.  
  12. j_setdebug(0)
  13. if( j_start() = J_FALSE ) then  
  14.    print "can't connect to JAPI server"    
  15.   end  
  16. endif  
  17.  
  18. indexbase 0
  19.  
  20. Dim int r(400*400), g(400*400), b(400*400)
  21. int gc
  22.  
  23. function julia(double x, y, r, s, int it) as double  
  24.   double dup
  25.   for k=0 to it
  26.      gc+=1
  27.      dup=x
  28.      x=x*x-y*y+r
  29.      y=2*dup*y+s
  30.      if abs(x*y)>4 then exit for
  31.   next
  32.   return k*abs(x*y)
  33. end function  
  34.  
  35. int m_p,it_p
  36. double xo_p, yo_P
  37. string color
  38. int col_val=2
  39.  
  40. sub initState()    
  41.     m_p=200 : it_p=12 : xo_p=-1.5 : yo_p=-1.5 : color="Green"  : col_val=2
  42. end sub
  43.  
  44. sub main()
  45.     double x , y , dz , rc , ic , xo , yo , orb
  46.  
  47.     sys frame, canvas
  48.      
  49.     frame=j_frame("Graphic Win - Julia Sets")
  50.     j_setflowlayout(frame,j_vertical)
  51.     j_setinsets(frame,50,20,20,20)
  52.     j_setnamedcolorbg(frame,j_light_gray)
  53.  
  54.     initState()    
  55.     toplbl=j_label(frame,"Julia params: m=" m_p ", it=" it_p ", xo=" xo_p ", yo=" yo_p ", color += " color)    
  56.     j_setnamedcolor(toplbl,j_blue)
  57.  
  58.     'Menu
  59.    menubar = j_menubar(frame)
  60.     file    = j_menu(menubar,"File")
  61.     save    = j_menuitem(file, "Save as BMP")
  62.     jprint   = j_menuitem(file,"Print")
  63.     quit    = j_menuitem(file,"Quit")
  64.  
  65.     calc    = j_menu(menubar,"Calc")
  66.     start   = j_menuitem(calc,"Start")
  67.     options = j_menuitem(calc, "Options")
  68.     reset   = j_menuitem(calc,"Reset")
  69.  
  70.     'Options Dialog
  71.    optdialog  = j_dialog(frame,"Options for Julia Sets")
  72.     j_setsize(optdialog,300,200)
  73.  
  74.     clr_lbl = j_label(optdialog, " Color:")
  75.     j_setpos(clr_lbl, 10,30)
  76.     radio =j_radiogroup(optdialog)
  77.     Red   = j_radiobutton(radio,"Red")
  78.     j_setpos(Red,20,60)
  79.     Green = j_radiobutton(radio,"Green")
  80.     j_setpos(Green,20,80)
  81.     Blue   = j_radiobutton(radio,"Blue")
  82.     j_setpos(Blue,20,100)
  83.    
  84.     p_lbl = j_label(optdialog,"Parameters:")
  85.     j_setpos(p_lbl, 150,30)
  86.     m_lbl = j_label(optdialog,"m:")
  87.     j_setpos(m_lbl, 150,60)
  88.     it_lbl = j_label(optdialog,"it:")
  89.     j_setpos(it_lbl, 150,80)
  90.     xo_lbl = j_label(optdialog,"xo:")
  91.     j_setpos(xo_lbl, 150,110)
  92.     yo_lbl = j_label(optdialog,"yo:")
  93.     j_setpos(yo_lbl, 150,130)
  94.     m_field = j_textfield(optdialog,5)
  95.     j_setpos(m_field, 180,60)    
  96.     it_field = j_textfield(optdialog,5)
  97.     j_setpos(it_field, 180,80)    
  98.     xo_field = j_textfield(optdialog,6)
  99.     j_setpos(xo_field, 180,110)    
  100.     yo_field = j_textfield(optdialog,6)
  101.     j_setpos(yo_field, 180,130)    
  102.  
  103.     close   = j_button(optdialog,"Ok")
  104.     j_setpos(close, 100,160)
  105.  
  106.     canvas=j_canvas(frame,400,400)
  107.     j_setnamedcolorbg(canvas,j_dark_gray)
  108.     lbl=j_label(frame,"INFO")
  109.     j_setsize(lbl,260,32)
  110.     j_setnamedcolor(lbl,j_blue)
  111.  
  112.     j_pack(frame)
  113.     j_setpos(frame, 200,100)
  114.     j_show(frame)
  115.  
  116.     obj=0
  117.     while obj != quit and obj != frame
  118.  
  119.        obj=j_getaction()
  120.                              
  121.        if obj=start then
  122.          j_settext(lbl,"Calculating - Please wait")
  123.  
  124.          gc=0
  125.          for m=m_p to 0 step -1
  126.             xo=xo_p
  127.             yo=yo_p
  128.             rc=0+m/100
  129.             ic=-1+m/100
  130.             dz=3/400
  131.             it=it_p
  132.             for i=0 to 399
  133.                x=xo+i*dz
  134.                for j=0 to 399
  135.                   y=yo+j*dz
  136.                   orb=julia(x,y,rc,ic,it)
  137.                   if color="Red" then   r(i+400*j)=int (5*orb)
  138.                   if color="Green" then g(i+400*j)=int (5*orb)
  139.                   if color="Blue" then  b(i+400*j)=int (5*orb)
  140.                next j
  141.             next i
  142.             j_drawimagesource(canvas,0,0,400,400,r,g,b)            
  143.          next m
  144.          j_settext(lbl,"MAIN ROUTINE CALLED "+str(gc) +" times" )
  145.          
  146.        end if
  147.  
  148.        if obj=reset then
  149.          initState()
  150.          j_settext(toplbl, "Julia params: m=" m_p ", it=" it_p ", xo=" xo_p ", yo=" yo_p ", color += " color)  
  151.          j_settext(lbl, "INFO")
  152.  
  153.          j_getimagesource(canvas,0,0,400,400,r,g,b)
  154.          for i=0 to <(400*400)
  155.             r[i]=0 : g[i]=0 : b[i]=0
  156.          next i            
  157.          j_drawimagesource(canvas,0,0,400,400,r,g,b)
  158.          j_setnamedcolorbg(canvas,j_dark_gray)        
  159.        end if
  160.  
  161.        if obj=jprint then j_print(canvas)
  162.  
  163.        if obj=options then
  164.          'place centered in frame        
  165.         x = j_getxpos(frame)+j_getwidth(frame)/2 - j_getwidth(optdialog)/2
  166.          y = j_getypos(frame)+j_getheight(frame)/2 -  j_getheight(optdialog)/2
  167.          j_setpos(optdialog,x,y)
  168.          'values
  169.         if col_val=1 then j_setstate(Red, j_true)
  170.          if col_val=2 then j_setstate(Green, j_true)
  171.          if col_val=3 then j_setstate(Blue, j_true)
  172.          j_settext(m_field, m_p)
  173.          j_settext(it_field, it_p)
  174.          j_settext(xo_field, xo_p)
  175.          j_settext(yo_field, yo_p)
  176.  
  177.          j_show(optdialog)
  178.         end if
  179.  
  180.         if obj=close or obj = optdialog then
  181.           j_hide(optdialog)
  182.           'update values
  183.          if j_getstate(Red)   = j_true then col_val=1 : color="Red"
  184.           if j_getstate(Green) = j_true then col_val=2 : color="Green"
  185.           if j_getstate(Blue)  = j_true then col_val=3 : color="Blue"
  186.           string tmp
  187.           tmp = j_gettext(m_field, tmp) : m_p=val(rtrim (ltrim (tmp)))
  188.           tmp = j_gettext(it_field, tmp) : it_p=val(rtrim (ltrim (tmp)))
  189.           tmp = j_gettext(xo_field, tmp) : xo_p=val(rtrim (ltrim (tmp)))
  190.           tmp = j_gettext(yo_field, tmp) : yo_p=val(rtrim (ltrim (tmp)))
  191.          
  192.           j_settext(toplbl, "Julia params: m=" m_p ", it=" it_p ", xo=" xo_p ", yo=" yo_p ", color += " color)
  193.         end if
  194.  
  195.         if obj=save then
  196.           image = j_image(400,400)
  197.           j_drawimagesource(image,0,0,400,400,r,g,b)          
  198.           if j_saveimage(image,"Julia.bmp",J_BMP) != 1 then
  199.             mbox "Error saving Bitmap file"
  200.           else
  201.             mbox "Saved as Julia.bmp"
  202.           end if  
  203.         end if
  204.  
  205.     wend
  206.     j_quit()
  207.  
  208. end sub
  209.  
  210. main()
  211.  


.
Title: Re: Japi Experiments
Post by: JRS on November 21, 2016, 10:19:13 AM
If JAPI wasn't a socket interface and supported the current Java UI API, it might be worth investing time in. IMHO

This also reminds me that I miss Rob's posts to the forum. I hope he is okay.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 21, 2016, 05:32:28 PM
John, Charles, gentlemen,

I'm afraid I'll have to bring in some sad news.

I did some quick search on the net with what little I knew about Rob outside this forum: he spoke Flemish, he was my coeval (born 1956), and apart from being a good programmer and mathematician, he was very proficient in photography and had a good command of related optical gear.

So what I found is this (https://www.flickr.com/photos/masi_1961/22844240889/in/photosof-32981029@N07/).

Rest in peace, friend. I value every moment we spent together on this forum. I'll be missing you for years to come.


 :'(
Title: Re: Japi Experiments
Post by: JRS on November 21, 2016, 08:53:24 PM
I will miss you Rob!



Title: Re: Japi Experiments
Post by: Charles Pegge on November 22, 2016, 01:33:52 AM
Very sorry to hear of Rob's early departure from us. Off into the grand fractal universe.

Here is some of Rob's valuable material from my desktop (mostly thinBasic)

.
Title: Re: Japi Experiments
Post by: Arnold on November 23, 2016, 01:29:09 AM
Yes, these are really sad news. As I started so late with Oxygen I often explore the forum for older messages and I found some lisp examples coded by him. Maybe they can be written in leanlisp or oxyscheme in some way too.

The julia example which I found with his message looks really nice and I wonder how the graphics could be influenced. I found some documentation about Julia fractals in internet but until now this seems to be a very complex topic.
Title: Re: Japi Experiments
Post by: Charles Pegge on November 23, 2016, 12:41:34 PM
Hi Roland,

Here it is without the JAPI dependency: I've tightened up the algorithm and run it in the ConsoleG framework, as a continuous animation.

Code: [Select]
  #compact
  % Title "Animated Julia Fractal Demo"
  % Animated
  % ScaleUp
 '% PlaceCentral
 '% AnchorCentral
 '% NoEscape

  includepath "$\inc\"
  '% filename "t.exe"
  'include "RTL64.inc"
  include "ConsoleG.inc"


  indexbase 0
  pixel4 juliapix[512*512]
  int gc
     
  function JuliaCalc(double x, y, r, s, int it) as double
  =======================================================
  double dup
  int k
  for k=0 to it
     gc+=1
     dup=x
     x=x*x-y*y+r
     y=2*dup*y+s
     if abs(x*y)>4 then exit for
  next
  return k*abs(x*y)
  end function 
     
  int m_p,it_p
  double xo_p, yo_P
  string colorstr
  int col_val=2
     

  sub JuliaInitState()
  ====================
  m_p=200 : it_p=12 : xo_p=-1.5 : yo_p=-1.5 : colorstr="Green"  : col_val=2
  end sub

  function JuliaRender(float mf,rf,gf,bf)
  =======================================
  'mf 2 .. 0 'factor
  'rf 0 .. 1 'red
  'gf 0 .. 1 'green
  'bf 0 .. 1 'blue
  sys i , j
  double x , y , dz , rc , ic , xo , yo , orb, amp
  gc=0
  pixel4 jp at @JuliaPix
  xo=xo_p
  yo=yo_p
  rc=mf
  ic=mf-1.0
  dz=3/512
  amp=5.0
  it=it_p
  for i=0 to <512
    y=yo+i*dz
    for j=0 to <512
      x=xo+j*dz
      orb=JuliaCalc(x,y,rc,ic,it)*amp
      jp.r=orb*rf
      jp.g=orb*gf
      jp.b=orb*bf
      jp.a=255
      @jp+=4 'next pixel
    next j
  next i
  end function


  function main()
  ===============
  '
  indexbase 1
  static int init
  static float jf,jfi
  if not init then
    JuliaInitState()
    CreateSynthTexture JuliaTex,JuliaPix,512*512*4
    jf=0
    jfi=0.002
    init=1
  end if
  JuliaRender jf,1,1,0
  jf+=jfi
  if jf<0.0 then
    jf=0.0 : jfi=-jfi
  elseif jf>2.0 then
    jf=2.0 : jfi=-jfi
  end if
  pushstate
    move 15,-15.0
    'UserMovement m2,200
    flat : color 1,1,1,1
    texture JuliaTex
    MakeTexture @JuliaPix, 512, 512, texn[JuliaTex] 'dynamic texture
    quadnorm 15.0,15.0 'apply image texture to quad
    texture 0
  popstate
  '
  end function



  EndScript

The next step would be to include some key-based controls, to be able to zoom in and pan around. :)
Title: Re: Japi Experiments
Post by: Arnold on November 24, 2016, 12:15:03 AM
Hi Charles,

your Julia demo works excellent. As it does contiuous animation there are always changing patterns. And the UserMovement macro is incredibly powerful. I can use arrow-keys (also with shift or control), page down, page up (also with shift or control), left/right/middle mouse button and move / turn  the object in all possible directions. This is not and will not be possible with the Japi library.

I have not yet figured out how the Julia algorithm works. But meanwhile I found that by changing the amp value it is already possible to create an infinite number of different patterns. I assume that by changing some other variables and manipulating the color values some more interesting effects are possible. This Julia algorithm is indeed amazing.

Roland
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 24, 2016, 05:23:19 AM
Hi Charles,

This is what I'm getting trying to run your Julia script with the latest Oxygen pack dated November 10, 2016 that I've just DL'ed from the Wizard.

I'm perfectly sure there's nothing of the kind located on my C:\ disk:

.
Title: Re: Japi Experiments
Post by: Arnold on November 24, 2016, 07:21:07 AM
Hi Mike,

I commented out in /inc/Sysutil.inc line 12: librarypath "$\" (only for this special purpose) and executed \tools\BuildCo2.bat. The resulting co2.exe will not include the path for oxygen.dll. You could also use co2.exe of the previous distribution.

Roland
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 24, 2016, 08:03:31 PM
Oh yes Roland,

That did the trick for me. Thanks! :)
Title: Re: Japi Experiments
Post by: Charles Pegge on November 24, 2016, 11:23:28 PM

Sorry about the 'baked' LibraryPath, Mike. Something to avoid in pre-compiled code!

Here is a slightly more efficient, FPUified version of the JuliaCalc function

Code: [Select]
  function JuliaCalc(double x, y, r, s, int it) as double
  =======================================================
  double xy,mx=4.0
  int k=it
  'for k=0 to it
  mov esi,it
  (
    dec esi
    jl exit
     'gc+=1
       inc dword gc
     'xy=x*y
       fld qword x
       fmul qword y
       fstp qword xy
     'x=x*x-y*y+r
       fld qword y
       fmul st0
       fld qword x
       fmul st0
       fsubp st1
       fadd qword r
       fstp qword x
     'y=2*xy+s
       fld qword xy
       fadd st0
       fadd qword s
       fstp qword y
     'if abs(xy)>mx then exit for
       fld qword mx
       fld qword xy
       fabs
       fcomip
       fstp st0
       ja exit
  'next
    repeat
  )
  'return k*abs(xy)
    fld qword xy
    fabs
    sub k,esi
    fimul dword k
    return
  end function 
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 25, 2016, 01:55:55 AM
Don't mention it, Charles! :)


Here is a slightly more efficient, FPUified version of the JuliaCalc function

Is that for the sake of avoiding the overhead inherent in HLL variable access? If yes than what's the estimated speed gain?
Title: Re: Japi Experiments
Post by: Charles Pegge on November 25, 2016, 09:19:40 AM
Just looking at the code, I would say there is a 12..15% saving due to reduced memory access. But cache memory is very fast.

You might be able to squeeze it further by using SIMD registers, and do most of the work on 3 pixels simultaneously :)
Title: Re: Japi Experiments
Post by: Arnold on November 25, 2016, 10:12:40 AM
Somehow this Julia is a bit seductive. I am still trying to improve the Japi demo and find out which params are important, but I already like the resulting images.

.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 25, 2016, 12:30:02 PM
Regretfully many if not all screenshots of Rob's fractals are now gone from the site (attachments are only kept for about two or three months and then deleted). But I remember that the best coloration attempts were made using Exp() and Log() combinations on the r, g, and b values, which unfortunately also made the calc much, much slower. Linear gradients like in these screenshots above are a little simplistic for this gorgeous fractal.

In fact trying to spectacularly colorize the dull insides of Julia was the very topic that made me acquainted with Rob on the thinBasic forum. :)
Title: Re: Japi Experiments
Post by: Charles Pegge on November 26, 2016, 03:40:30 AM
We could use GLSL pixel shaders for high performance rendering of fractals, subject to the limited precision of floats.

http://n.ethz.ch/~vartokb/gl-shaders.html
-->
glsl fractals
https://www.youtube.com/watch?v=H-3yPXXZUC8
-->
Best fractals zoom ever
https://www.youtube.com/watch?v=BTiZD7p_oTc
-->
**
Mandelbrot Zoom 10^225 1080p - Grand Pianola Music
https://www.youtube.com/watch?v=n9nf5ShnRfk
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 26, 2016, 11:15:40 AM
Charles,

Thanks for the links!

This glossy, silky, shiny coloration is what I was talking about mentioning Exp() and Log():

(http://i1240.photobucket.com/albums/gg490/FbslGeek/Glossy.png)

It's relatively easy to apply to Julia's curves but not so, to its plain insides. The insides are in fact also structured but their variations fall outside the scope of calc in the normally renderable parts of the fractal. The variations aren't however as rich nor as pronounced:

(http://i1240.photobucket.com/albums/gg490/FbslGeek/DuoMandel.png)
Title: Re: Japi Experiments
Post by: Charles Pegge on November 27, 2016, 01:04:48 PM
I think byte overflow coloration, as in Roland's example here, is quite a good way to increase the sensitivity in the interior.

I've attached most of the Julia variables to the keyboard, and get a wide variety of views:



.
Title: Re: Japi Experiments
Post by: Charles Pegge on November 27, 2016, 01:10:54 PM

I wonder why 3d fractals are so spooky:

Trip inside a 3D fractal (Kleinian) GPU realtime rendering
https://www.youtube.com/watch?v=XIzScwydxOE
-->
Like in a dream - 3D fractal trip
https://www.youtube.com/watch?v=S530Vwa33G0
-->
Like in a dream - 3D fractal trip
https://www.youtube.com/watch?v=HCl-7AbyCsY
-->
Amazing 3D Fractal Compilation + 4D with Jazzy Groove House Mix
https://www.youtube.com/watch?v=LbWS79Cxykg
Title: Re: Japi Experiments
Post by: Arnold on November 28, 2016, 03:10:52 AM
Hi Charles, Mike,

I am lightyears away from your level. The more I learn by / with Oxygenbasic the more I feel like Sokrates (I know that I know nothing). In my earlier life I counted many things but neither pixels nor complex numbers. And now I am sitting here and try to remember about trigonometric functions, differentiation, derivation rules, find out the real and imaginary parts of life. But I learned that I only use 10 percent of my brain so maybe there is still some hope.

Until now I was only aware of some basic geometric shapes and the basic Mandelbrot image. I never had the idea that there is so much possible by applying some sophisticated formulae in a computer program.

Roland

Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 28, 2016, 09:13:11 AM
Hi Charles, Roland,


I wonder why 3d fractals are so spooky:

No they aren't for me. They are terrifically awesome, hypnotic, and viral. :)


... byte overflow coloration ...

Your renders are beautiful, Charles. But byte range is too narrow and restrictive to render the entire of a Julia, hence exponents and logarithms.

What I really mean by Julia insides is seen in your bottommost violet snapshot as black areas. The central regions clamped to black (or to white in some other programs) by byte linear coloration are in fact full of life of their own. Here's Rob's best attempt to colorize the black abyss inside a 2D Julia fractal in real time:

(http://i1240.photobucket.com/albums/gg490/FbslGeek/Interference.png)

... I feel like Sokrates (I know that I know nothing). In my earlier life I counted many things but neither pixels nor complex numbers. And now I am sitting here and try to remember about trigonometric functions, differentiation, derivation rules, find out the real and imaginary parts of life. But I learned that I only use 10 percent of my brain so maybe there is still some hope.

I couldn't put it better than you did, Roland! :D
Title: Re: Japi Experiments
Post by: Charles Pegge on November 29, 2016, 01:22:53 AM

Interiors using sqr(abs(x)) and sqr(sqr(abs(x))) as the output of JuliaCalc:

It is useful to have full control of the iterations, and output multiplier to get the best visual:





.
Title: Re: Japi Experiments
Post by: Charles Pegge on November 29, 2016, 09:09:56 AM
abs(x) is used to render yellow and  abs(y) renders blue:



.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 29, 2016, 12:37:01 PM
WOW!!!

This is all VERY impressive indeed, Charles!

And those Julias remind me strongly of a pair of Italian (Peninsula) boots! :D
Title: Re: Japi Experiments
Post by: Arnold on November 30, 2016, 03:35:07 AM
Hi Charles,

you changed JuliaCalc and JuliaRender a little bit? I do not know the meaning of sc, and I suddenly see Mandelbrot. I can achieve some effects but they are far away from your results. Your images appear like some kind of artwork - ars mathematice.

Would it be possible to adapt the julia algorithm for different sizes? (no streching). If I change width or height I only get garbage. The best result would be if I use width = 2*height but then I only get the picture twice.

This is my trial with Japi (of course not comparable with Opengl)

Roland

Code: [Select]
'Using the Japi library
'www.japi.de

$ filename "julia2.exe"
'include "$/inc/RTL32.inc"
'include "$/inc/console.inc"
include "$/inc/dynamic.inc"

extern lib "$/projectsC/Japi/japi.dll" cdecl
include once "$/projectsC/Japi/japi.h"
end extern

j_setdebug(0)
if( j_start() = J_FALSE ) then 
   print "can't connect to JAPI server"     
   end 
endif 

indexbase 0

int c_width = 800
int c_height= 400

int mp  = 40
int itp =  2
double xop  = -1.5
double yop  = -1.5
double amp_p = 8.0

double crf = 0.3    'red factor
double cgf = 0.4    'green
double cbf = 0.4    'blue

'sc ?

DynDim int r(c_width*c_height) : DynDim int g(c_width*c_height) : DynDim int b(c_width*c_height)
int gc

function JuliaCalc(double x, y, r, s, int it) as double 
  double mx=4.0

  for k=0 to it
     gc+=1
     xy=x*y
     x=x*x-y*y+r
     y=2*xy+s
     if abs(xy)>mx then exit for
  next
  return k*abs(xy)
end function 

function JuliaRender(int m, canvas,double rf,gf,bf)
==================================================
  'mf        'loop factor
  'rf 0 .. 1 'red
  'gf 0 .. 1 'green
  'bf 0 .. 1 'blue
  double mf=m/100
  sys i , j
  double x , y , dz , rc , ic , xo , yo , orb, amp

  rf*=2.55 : gf*=2.55 : bf*=2.55
   
  gc=0
  xo=xop
  yo=yop
  rc=mf
  ic=mf-1.0
  dz=3/c_width
  amp=amp_p
  it=itp
  for i=0 to <c_height
    y=yo+i*dz 
    for j=0 to <c_width
      x=xo+j*dz
      orb=JuliaCalc(x,y,rc,ic,it)*amp
      idx=i+c_height*j     
      r(idx)=rf*orb
      g(idx)=gf*orb
      b(idx)=bf*orb
    next j
  next i
  j_drawimagesource(canvas,0,0,c_width,c_height,r,g,b)
end function


sub main()

    sys frame, canvas
   
    frame=j_frame("Graphic Win - Julia Sets")
    j_setflowlayout(frame,j_vertical)
    j_setinsets(frame,50,20,20,20)
    j_setnamedcolorbg(frame,j_light_gray)
   
    toplbl=j_label(frame,"Julia params: m=" mp ", it=" itp ", xo=" xop ", yo=" yop)   
    j_setnamedcolor(toplbl,j_blue)

    'Menu
    menubar = j_menubar(frame)
    file    = j_menu(menubar,"File")
    save    = j_menuitem(file, "Save as BMP")
    jprint   = j_menuitem(file,"Print")
    quit    = j_menuitem(file,"Quit")

    calc    = j_menu(menubar,"Calc")
    start   = j_menuitem(calc,"Start")

    canvas=j_canvas(frame,c_width,c_height)
    j_setnamedcolorbg(canvas,j_dark_gray)
    lbl=j_label(frame,"INFO")
    j_setsize(lbl,260,32)
    j_setnamedcolor(lbl,j_blue)

    j_pack(frame)
    j_setpos(frame, 200,100)
    j_show(frame)

    obj=0
    while obj != quit and obj != frame

       obj=j_getaction()
                             
       if obj=start then
         j_settext(lbl,"Calculating - Please wait")

         for m=mp to 0 step -1
            JuliaRender m, canvas, crf, cgf, cbf
         next m
         j_drawimagesource(canvas,0,0,c_width,c_height,r,g,b)           

         j_settext(lbl,"MAIN ROUTINE CALLED "+str(gc) +" times" )
         
       end if

       if obj=jprint then j_print(canvas)

       if obj=save then
          image = j_image(c_width,c_height)
          j_drawimagesource(image,0,0,c_width,c_height,r,g,b)           
          if j_saveimage(image,"Julia.bmp",J_BMP) != 1 then
            mbox "Error saving Bitmap file"
          else
            mbox "Saved as Julia.bmp"
          end if 
       end if

    wend
    DynFree r : DynFree g : DynFree b   
    j_quit()

end sub

main()

.
Title: Re: Japi Experiments
Post by: Charles Pegge on November 30, 2016, 08:41:45 AM
Hi Roland, I am using an interactive derivation, JuliaKeyb, which has become a laboratory for testing all sorts of variations.

+/-  increase/decrease
arrow keys to pan
z  to zoom
f   for julia factor
i   for iterations
m  for color multiplier
j   to toggle julia/mandelbrot

r g b for colors

the program is directly altered for all other explorations.

Today's code:

Code: [Select]
  #compact
  % Title "Fractal Demo: keyboard: + - r g b f i j m  Ctrl P for snapshot"
 '% Animated
  % ScaleUp
 '% PlaceCentral
 '% AnchorCentral
 '% NoEscape

  includepath "$\inc\"
  '% filename "t.exe"
  'include "RTL64.inc"
  include "ConsoleG.inc"

  macro limit(v,min,max)
  ======================
  if v>max then
    v=max
  elseif v<min then
    v=min
  end if
  end macro

  indexbase 0
  pixel4 juliapix[512*512]
  int gc
     
  function JuliaCalc(double x, y, r, s,*oc, int it,jm) as double
  ==============================================================
  double xy,mx=4.0
  int k=it
  '
  if jm then
    'SWITCH TO MANDELBROT SET
    r=x
    s=y
    x=0
    y=0
  end if
  '
  'for k=0 to it
  mov esi,it
  (
    dec esi
    jl exit
     'gc+=1
       inc dword gc
     'xy=x*y
       fld qword x
       fmul qword y
       fstp qword xy
     'x=x*x-y*y+r
       fld qword x
       fmul st0
       fld qword y
       fmul st0
       fsubp st1
       fadd qword r
       fstp qword x
     'y=2*xy+s
       fld qword xy
       fadd st0
       fadd qword s
       fstp qword y
     'if abs(xy)>mx then exit for
       'fld qword mx
       'fld qword xy
       'fabs
       'fcomip
       'fstp st0
       'ja exit
     'if x*x+y*y>mx then exit for
       fld qword mx
       fld qword x
       fmul st0
       fld qword y
       fmul st0
       faddp st1
       fcomip
       fstp st0
       ja exit
  'next
    repeat
  )
 oc={x,y}
 'return sqr(sqr(abs(y)))
 'return sqr(abs(x)+abs(y))
 'return sqr(abs(y))
  return abs(x)
 'return x+y
 'return abs(x)+abs(y)
 'return abs(xy)
 'return hypot(x,y)
  '
  sub k,esi
  'return k*abs(xy)
    'fld qword xy
    'fabs
    'fimul dword k
    'return
  'return abs(xy)
    'fld qword xy
    'fabs
    'return
  'return xy
    'fld qword xy
    'return
  'return xy*xy
    'fld qword xy
    'fmul st0
    'return
  'return k
  end function 
     
     


  function JuliaRender(double mf,it,jm,mp,sc,xo,yo,rf,gf,bf)
  ==========================================================
  indexbase 0
  'mf 2 .. 0 'factor
  'rf 0 .. 1 'red
  'gf 0 .. 1 'green
  'bf 0 .. 1 'blue
  sys i , j
  double x , y , dz , rc , ic , orb, oc[1]
  gc=0
  pixel4 jp at @JuliaPix
  rc=mf
  ic=mf-1.0
  dz=sc*3/512
  for i=0 to <512
    y=yo-1.5*sc+i*dz
    for j=0 to <512
      x=xo-1.5*sc+j*dz
      orb=JuliaCalc(x,y,rc,ic,oc,it,jm)*mp
      '
      'RECIPROCALS
      if oc[0] then oc[0]=1/abs(oc[0])
      if oc[1] then oc[1]=1/abs(oc[1])
      jp.r=abs(oc[0])*mp*rf
      jp.g=abs(oc[0])*abs(oc[1])*mp*gf
      jp.b=abs(oc[1])*mp*bf
     'jp.r=orb*rf 'red
     'jp.g=orb*gf 'green
     'jp.b=orb*bf 'blue
      jp.a=255    'alpha
      @jp+=4      'next pixel
    next j
  next i
  end function


  function main()
  ===============
  '
  indexbase 1
  static int init,it,jm,mp
  static double rf,gf,bf,jf,jfi,sc,xo,yo
  if not init then
    sc=1
    xo=0
    yo=0
    rf=1.0
    gf=.5
    bf=.2
    jf=0
    jfi=0.002
    it=10
    mp=5
    init=1
    NewTexture JuliaTex
  end if
  if lastkey or lastchar then
    static float kvi=.01, kva=0.05
    static int   kii=1
    int cey=lastchar
    kva=.05*sc
    if cey>0x60 and cey<0x7b then cey-=32 'to uppercase
    '
    if key[189]      then kvi=-.01 : kii=-1 '-'
    if key[187]      then kvi= .01 : kii= 1 '+'
    if key[VK_RIGHT] then xo+=kva           'ARROW RIGHT'
    if key[VK_LEFT]  then xo-=kva           'ARROW LEFT'
    if key[VK_UP]    then yo+=kva           'ARROW UP'
    if key[VK_DOWN]  then yo-=kva           'ARROW DOWN'
    if key[0x5A]     then sc*=(1-kvi*5)     'Z' ZOOM
    if key[0x46]     then jf+=kvi           'F' FACTOR
    if cey=0x49      then it+=kii           'I' ITERATIONS
    if cey=0x4a      then jm=1-jm           'J' JULIA/MANDELBROT MODE
    if cey=0x4d      then mp+=kii           'M' OUTPUT MULTIPLIER
    if key[0x52]     then rf+=kvi           'R' RED
    if key[0x47]     then gf+=kvi           'G' GREEN
    if key[0x42]     then bf+=kvi           'B' BLUE
    '
   'limit sc, .01,100 'Julia Zoom
   'limit jf, 0,2     'Julia factor
    limit it, 1,1000  'JuliaCalc iterations
    limit mp, 1,1000  'Julia output multiplier
    limit rf, 0,1     'red
    limit gf, 0,1     'green
    limit bf, 0,1     'blue
    '
    '
    lastkey=0
    lastchar=0
  end if
  '
  'DISPLAY FACTORS
  def fi "  %1:" str(%1,0)
  def ff "  %1:" str(%1,2)
  def ff3 "  %1:" str(%1,3)
  printl ff jf fi it  fi mp  ff3 sc ff3 xo ff3 yo ff rf  ff gf  ff bf
  '
  JuliaRender jf,it,jm,mp,sc,xo,yo,rf,gf,bf
  'jf+=jfi
  'if jf<0.0 then
  '  jf=0.0 : jfi=-jfi
  'elseif jf>2.0 then
  '  jf=2.0 : jfi=-jfi
  'end if
  pushstate
    move 15,-15.0
    'UserMovement m2,200
    flat : color 1,1,1,1
    texture JuliaTex
    DynSynthTexture JuliaTex,JuliaPix,512*512*4 
    quadnorm 15.0,15.0 'apply image texture to quad
    texture 0
  popstate
  '
  end function



  EndScript

Title: Re: Japi Experiments
Post by: Charles Pegge on November 30, 2016, 08:51:36 AM
Mike,

More interior revelations :)

Using 1/abs(x) and 1/abs(y) and their roots  to colorize:




.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on November 30, 2016, 09:09:22 AM
OMG Charles!

!!! THIS IS BEAUTIFUL !!!

You simply MUST make the texture twice larger, ensure it renders at a 1:1 scale, and save your renders in a lossless PNG format!

I think you could also use fast integer square root approximations to compensate for the larger texture sizes. It shouldn't affect the coloration quality to any visible extent, I think.



P.S. Your Mandelbrot is simply AWESOME! :)
Title: Re: Japi Experiments
Post by: Arnold on December 01, 2016, 12:58:43 AM
Hi Charles,

it is impressive to see how you improved and enhanced the basic code. I will not continue to try this with Japi. The use of Japi as a gui framework is limited, the graphical capabilities can not be compared with Opengl and javaw.exe will consume a lot of memory. So I think this would be a waste of time and resources. Maybe I will try to use JuliaCalc and JuliaRender with Freeglut and MinWin. The purpose then would be to learn a bit more about Opengl.

Roland
Title: Re: Japi Experiments
Post by: Arnold on December 01, 2016, 01:01:51 AM
Hello,

I almost forgot about my main purpose for this topic. These are the C-examples which I found at www.japi.de ported to Oxygenbasic, modified a little bit sometimes. Some demos should be revised a little bit more. My main purpose here was to check how much of the C-code could be used unmodified. (about 90%) It should be no problem to convert the remaining code to nice Oxygen syntax.

The examples should run anywhere on disk or usb-stick. I tested them with Win 32-bit and Win 64-bit. The file japi.inc expects Japi.dll and japi.h in folder projectsC. If using a previous distribution of OxygenBasic then projectsC must be changed to projectsB in japi.inc

Although the examples run quite ok, there are some issues with the japi library:
using some japi commands incorrectly will cause a program to freeze and it will not terminate correctly. Closing an application via the debug window can also prevent gxo2.exe / co2.exe from terminating. I often had to use the task manager to kill a process. Sometimes there is also a problem with user access control. As japi calls Java there is much memory used with javaw.exe. And last but least there are many missing controls which would be necessary to create a more ambitious application.

Nevertheless it was very interesting to see that OxygenBasic had no problem with running Japi. No wrapping at all like in many other programming languages.


.
Title: Re: Japi Experiments
Post by: JRS on December 01, 2016, 01:46:59 AM
Quote
Nevertheless it was very interesting to see that OxygenBasic had no problem with running Japi. No wrapping at all like in many other programming languages.

As I mentioned before, JAPI is a socket interface to the JVM running in a JAR.

Title: Re: Japi Experiments
Post by: Charles Pegge on December 01, 2016, 09:19:58 AM
Mike, these are 800x800, 100% quality png, sample sampled from a 1024x1024 texture.



.
Title: Re: Japi Experiments
Post by: Charles Pegge on December 01, 2016, 09:37:10 AM

JAPI seems to require an internet connection, at least on my PC, which likes to isolate itself occasionally.
Title: Re: Japi Experiments
Post by: Arnold on December 01, 2016, 12:30:12 PM
Hi Charles,

this really scared me and I immediately disconnected my pc from the net. But the demos worked ok. (sigh). So if an app using japi seems not to work maybe there is a messagebos or console window iconized in the taskbar. When porting the provided demos I sometimes found that some improper used statements could hang the program and I had to use taskmanager to kill the process of gxo2.exe / co2.exe.

I only used the java installation as provided by Oracle. I did not need a folder C:/JRE as stated in japi.de.
If I understand it correctly the used classes of the specified jar file in the developer's homepage are coded into japi.dll and there are some functions provided to call the classes, properties and methods. This could be interesting to learn some principles.

But I now think it would be more effective to learn Java and use and extend the classes directly.

Roland
Title: Re: Japi Experiments
Post by: JRS on December 01, 2016, 02:45:40 PM
Quote
this really scared me and I immediately disconnected my pc from the net.

Don't be paranoid!

JAPI embeds a binary image of a JAR running a socket connection (server) to communicate with Windows.  Look at the code if you don't believe me.

Quote
But I now think it would be more effective to learn Java and use and extend the classes directly.

You should try B4J (https://www.b4x.com/b4j.html). (BASIC for Java) It's FREE and is done by the same author that does the commercial B4A. (BASIC for Android)



Title: Re: Japi Experiments
Post by: Charles Pegge on December 02, 2016, 01:07:36 AM
Chocolate Aero Delight: :)



.
Title: Re: Japi Experiments
Post by: Arnold on December 02, 2016, 01:31:40 AM
Charles,

this is another very impressive view. Did you use zooming?

Roland
Title: Re: Japi Experiments
Post by: Arnold on December 02, 2016, 01:34:02 AM
Hi John,

thank you for clarification. Sometimes I am very vague and inaccurate.

Roland
Title: Re: Japi Experiments
Post by: Charles Pegge on December 02, 2016, 08:42:28 AM
Hi Roland,

Yes, zooming in around 3000x. You can see the approx coords xo,yo and other metrics on this similar snapshot.

Photo quality function, ctrl-Q included



.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on December 02, 2016, 03:32:11 PM
Great! :D

Two more animated sweeties for your choco table using FBSL's standard include class COGLApp v1.1 (~50KB of raw code) that also handles GLSL.

If you have a nVidia GTX series video card then the apps (see the attached zip) shouldn't use more than 0.5% of your CPU though of course your GPU usage may rise up to 70..80% when the app windows are maximized:

.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on December 02, 2016, 04:09:43 PM
Don't be paranoid!

John,

Well behaved MS Windows apps are expected to inform their users that they are going to use net interfaces/protocols for whatever purposes. Otherwise their behavior is presumed suspicious and ill-intended on default. Hence Charles' and Roland's natural base-instinctive actions that I can only share and approve of.

MS Windows offers a wide variety of native system facilities to ensure Win32/64 apps' interaction with their constituent modules, and using sockets for the apps' intrinsic needs is a sheer oddity. Probably you're used to this in your alien Andr-omed-oid environments but we -- in MS Windows -- aren't.

I thought we elaborated on that Java issue exhaustively when Rob was still around. Let Java live in my browsers once it's built in them as well as in the net itself by joint *nixoid efforts but that's about as far as Java is going to ever get on my machines.
Title: Re: Japi Experiments
Post by: JRS on December 02, 2016, 06:03:08 PM
My only point was that it was an internal socket connection and not a call to mother Oracle.

The other point I tried to make is JAPI has been dead for years and not using the modern Swing UI. I gave up on it years ago,
Title: Re: Japi Experiments
Post by: Charles Pegge on December 04, 2016, 01:37:13 AM

Thanks Mike, I look forward to analysing your glsl chocolate delights :)

Paired Spirals in the SeaHorse valley:



.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on December 04, 2016, 07:17:40 AM
I look forward to analysing your glsl chocolate delights :)

Absolutely no mystery there. The entire FBSL code is just an OpenGL canvas that only renders the final quad whose pixels are colorized by the fragment shader. It also supplies the shader with just two uniforms -- a vec2 with current horz and vert sizes of the canvas, and a float time in seconds (accurate to three decimal places) since the app started.

On default, the GLSL engine applies the entire fragment shader sequentially to each pixel in its intrinsic gl_FragCoord.xy or gl_TexCoord.xy range and yields the final color of this very pixel as the result of complex transformations that you see in this fragment shader. Neither C nor Asm can compete on the 4 to 8 CPU cores with GLSL speed (though fragment shader code is relatively easy to port to at least ANSI C) because GLSL uses hundreds if not thousands of cores available on modern (nVidia) GTX GPUs for parallel processing in real time, and even so GLSL programs may be difficult for your PC to render (see below).

Back to our OP Julias. What you see below is an animated 3D Juliabulb (cf. Mandelbulb) GLSL program. As usual for Íñigo Quílez (http://www.iquilezles.org/) with his ray marching techniques, the vertex shader is nearly empty while the fragment shader is a killer. :)

.
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on December 05, 2016, 04:57:42 AM
Hi guys,

I realize that my FBSL submissions might be regarded as spam but what I upload is really very basic stuff that can be easily re-implemented in Oxygen. I could've done it myself but I'm not as proficient in O2 as I am in FBSL and I just wouldn't like to lose time stumbling around where a seasoned fresh Oxygen breather can get things done in minutes.

What FBSL does with these shaders is this:

Code: Script BASIC
  1. '----------------------------------------
  2. ' -=:: GLSL Julia Demo for FBSL v3.5 ::=-
  3. '----------------------------------------
  4. ' #AppType Console ' for debugging only
  5. #Option Strict
  6. #DllImports OpenGL32
  7.  
  8. #Include <Include\Windows.inc>
  9. #Include ".\COGLApp1.inc"
  10.  
  11. #Define GL_QUADS 0x0007
  12. #Define GL_VERTEX_SHADER 0x8B31
  13. #Define GL_FRAGMENT_SHADER 0x8B30
  14. #Define GL_MODELVIEW 0x1700
  15. #Define GL_PROJECTION 0x1701
  16. #Define GL_COMPILE_STATUS 0x8B81
  17. #Define GL_LINK_STATUS 0x8B82
  18. #Define GL_INFO_LOG_LENGTH 0x8B84
  19.  
  20. ' GLSL Stubs for wglGetProcAddress
  21. Declare glAttachShader()
  22. Declare glCompileShader()
  23. Declare glCreateProgram()
  24. Declare glCreateShader()
  25. Declare glDeleteProgram()
  26. Declare glDeleteShader()
  27. Declare glGetProgramInfoLog()
  28. Declare glGetProgramiv()
  29. Declare glGetShaderInfoLog()
  30. Declare glGetShaderiv()
  31. Declare glGetUniformLocation()
  32. Declare glLinkProgram()
  33. Declare glShaderSource()
  34. Declare glUseProgram()
  35. Declare glUniform1f()
  36. Declare glUniform2f()
  37.  
  38. Dim width, height, Viewer
  39. Dim juliaVS = 0, juliaFS = 0, juliaProg = 0
  40. Dim unitime = 0, unires = 0
  41.  
  42. Sub Main()
  43.   Viewer = New COGLApp1("-=:: GLSL Julia Particles Demo ::=-", _
  44.   AddressOf Renderer, AddressOf Messenger, AddressOf Updater)
  45.  
  46.   glslInit()
  47.   Init()
  48.  
  49.   With Viewer
  50.     .DataRate(2)
  51.     Center(.hWnd()): Show(.hWnd)
  52.     .Start(): Delete Viewer
  53.   End With
  54. End Sub
  55.  
  56. Sub Updater()
  57.   Viewer.Caption("-=:: GLSL Julia Particles Demo ::=-   FPS = " & Viewer.FPS())
  58. End Sub
  59.  
  60. Function Messenger(ByVal %hwnd, ByVal %msg, ByVal %wparam, ByVal %lparam) As Long
  61.   If msg = WM_KEYDOWN Then
  62.     If wparam = VK_ESCAPE Then
  63.       Viewer.Quit()
  64.       Deinit()
  65.     End If
  66.     Return 0 ' suppress default
  67.  ElseIf msg = WM_SIZE Then
  68.     width = LoWord(lparam)
  69.     height = HiWord(lparam)
  70.     Reshape()
  71.     Return 0 ' ditto
  72.  ElseIf msg = WM_MOUSEWHEEL Or msg = WM_MOUSEMOVE Then
  73.     Return 0 ' ditto
  74.  ElseIf msg = WM_CLOSE Then
  75.     Deinit()
  76.   End If
  77.   Return -1 ' allow COGLApp's default processing
  78. End Function
  79.  
  80. Sub Renderer()
  81.   glPushMatrix(): glLoadIdentity()
  82.   glMatrixMode(GL_PROJECTION)
  83.   glPushMatrix(): glLoadIdentity()
  84.   glOrtho(0.0, 1.0, 0.0, 1.0, -1.0, 1.0)
  85.  
  86.   glUseProgram(juliaProg)
  87.  
  88.   glUniform1f(unitime, !Clock())
  89.   glUniform2f(unires, !width, !height)
  90.  
  91.   glBegin(GL_QUADS)
  92.   glVertex2i(0, 0)(1, 0)(1, 1)(0, 1) ' FBSL shorthand
  93.  glEnd()
  94.  
  95.   glUseProgram(0)
  96.  
  97.   glPopMatrix()
  98.   glMatrixMode(GL_MODELVIEW)
  99.   glPopMatrix()
  100. End Sub
  101.  
  102. Sub Init()
  103.   juliaVS = CompileFromFile(GL_VERTEX_SHADER, ".\JP.vs")
  104.   juliaFS = CompileFromFile(GL_FRAGMENT_SHADER, ".\JP.fs")
  105.   juliaProg = LinkProgram(2, juliaVS, juliaFS)
  106.  
  107.   glUseProgram(juliaProg)
  108.   unitime = glGetUniformLocation(juliaProg, "iGlobalTime")
  109.   unires = glGetUniformLocation(juliaProg, "iResolution")
  110.   glUseProgram(0)
  111. End Sub
  112.  
  113. Sub Deinit()
  114.   glDeleteProgram(juliaProg)
  115. End Sub
  116.  
  117. Sub Reshape()
  118.   Dim fov = 45., aspect = width / height
  119.  
  120.   glViewport(0, 0, width, height)
  121.   glMatrixMode(GL_PROJECTION)
  122.   glLoadIdentity()
  123.   fov = Tan(fov * PI / 360) * .1: aspect = aspect * fov
  124.   glFrustum(-aspect, aspect, -fov, fov, .1, 1000.)
  125.   glMatrixMode(GL_MODELVIEW)
  126. End Sub
  127.  
  128. Sub LoadSource(filePath) ' returns shader source in FileGet
  129.  FileGet(FileOpen(filePath, BINARY), FileLen(filePath))
  130.   FileClose(FileOpen) ' FBSL shorthand
  131. End Sub
  132.  
  133. Function CompileFromFile(enumtype As Long, filePath) As Long
  134.   Dim pointer, shader
  135.   Dim length As Long, result As Long
  136.  
  137.   LoadSource(filePath)
  138.   If Not Asc(FileGet) Then Return 0
  139.  
  140.   shader = glCreateShader(enumtype)
  141.   length = StrLen(FileGet)
  142.   pointer = @FileGet
  143.   glShaderSource(shader, 1, @pointer, @length)
  144.   glCompileShader(shader)
  145.   FileGet = "" ' optionally reclaim memory
  146.  
  147.   glGetShaderiv(shader, GL_COMPILE_STATUS, @result)
  148.   If Not result Then
  149.     Dim prglog As String
  150.    
  151.     glGetShaderiv(shader, GL_INFO_LOG_LENGTH, @length)
  152.     Alloc(prglog, length)
  153.     glGetShaderInfoLog(shader, length, @result, @prglog)
  154.    
  155.     Print "CompileFromFile(): Unable to compile ", filePath, ": ", prglog
  156.     Clean prglog
  157.    
  158.     glDeleteShader(shader)
  159.     Return 0
  160.   End If
  161.  
  162.   Return shader
  163. End Function
  164.  
  165. Function LinkProgram(numShaders, ParamArray shaders) As Long
  166.   Dim program = glCreateProgram(), shader
  167.  
  168.   For Dim i = 0 To numShaders - 1
  169.     shader = shaders[i]
  170.     glAttachShader(program, shader)
  171.   Next
  172.  
  173.   glLinkProgram(program)
  174.  
  175.   Dim success = 0
  176.   glGetProgramiv(program, GL_LINK_STATUS, @success)
  177.  
  178.   If Not success Then
  179.     Dim $temp * 256
  180.     glGetProgramInfoLog(program, 256, NULL, @temp)
  181.     Print "Failed to link program: ", temp
  182.     glDeleteProgram(program)
  183.     program = 0
  184.   End If
  185.  
  186.   Return program
  187. End Function
  188.  
  189. Sub GlslInit()
  190.   AddressOf glAttachShader = wglGetProcAddress("glAttachShader")
  191.   AddressOf glCompileShader = wglGetProcAddress("glCompileShader")
  192.   AddressOf glCreateProgram = wglGetProcAddress("glCreateProgram")
  193.   AddressOf glCreateShader = wglGetProcAddress("glCreateShader")
  194.   AddressOf glDeleteProgram = wglGetProcAddress("glDeleteProgram")
  195.   AddressOf glDeleteShader = wglGetProcAddress("glDeleteShader")
  196.   AddressOf glGetProgramInfoLog = wglGetProcAddress("glGetProgramInfoLog")
  197.   AddressOf glGetProgramiv = wglGetProcAddress("glGetProgramiv")
  198.   AddressOf glGetShaderInfoLog = wglGetProcAddress("glGetShaderInfoLog")
  199.   AddressOf glGetShaderiv = wglGetProcAddress("glGetShaderiv")
  200.   AddressOf glGetUniformLocation = wglGetProcAddress("glGetUniformLocation")
  201.   AddressOf glLinkProgram = wglGetProcAddress("glLinkProgram")
  202.   AddressOf glShaderSource = wglGetProcAddress("glShaderSource")
  203.   AddressOf glUseProgram = wglGetProcAddress("glUseProgram")
  204.   AddressOf glUniform1f = wglGetProcAddress("glUniform1f")
  205.   AddressOf glUniform2f = wglGetProcAddress("glUniform2f")
  206. End Sub

where Viewer is an instance of FBSL OpenGL application, and Messenger(), Renderer() and Updater() are its callbacks for handling the Windows message pipe, ensure timely OpenGL rendering, and  perform auxiliary tasks like e.g. display the FPS rate, respectively.

Similar things can be easily set up in standard Oxygen Basic to use the same shaders without modification.

I've just found yet another awesome shader that's strictly in line with the current topic and is IMHO even more beautiful than the notorious Julia Rings by Relsoft. It also shows that our perception of Julia fractal as a flat 2D pattern is too simplistic; in fact Julia has a very pronounced z-axis component, and the particles roller coasting below are a very nice visualization of that fact.

I took every effort to make the shader as light as possible, and it shouldn't stress your GTX GPU more than 50% (CPU <= .5%) @60FPS full screen while older geForce GPUs may experience a drop to 30FPS when the window is maximized.

Feel free to experiment with the shape and number of particles as well as with other arguments of the calc but keep an eye on your GPU load and temperature. I wouldn't want to see you burning down your wives' Amazon/eBay surfing gear. ;)

Enjoy! :)

.
Title: Re: Japi Experiments
Post by: Charles Pegge on December 06, 2016, 01:53:56 AM
Hi Mike,

A stunningly excellent find!

and thanks also for the FBSL source. GLSL setup is complex, and I am trying to establish a framework, robust enough to carry the burden, without being too restrictive.
Title: Re: Japi Experiments
Post by: Arnold on December 09, 2016, 03:56:22 AM
Hi Charles,

I was prevented from my pc the last few days. Yesterday I tried out your JuliaKeyb demo, but two subs/macros/structs(?) were reported as missing: NewTexture and DynSynthTexture. I assume you added these two items in ConsoleG.inc or in one of the other dependant include files? What will be the purpose of these two routines?

Roland
Title: Re: Japi Experiments
Post by: Arnold on December 09, 2016, 03:57:29 AM
Hi Mike,

some time ago I installed FBSL 3.40.10 to learn more about the language. Is it possible to get the latest FBSL v3.5 which will run your provided FBSL Julia Demo code? In the FBSL forum I found the link for FBSL v3.5 RC2.zip. Is this the required package?

Roland
Title: Re: Japi Experiments
Post by: Mike Lobanovsky on December 09, 2016, 07:08:05 PM
Hi Roland,

Yes, if the older v3.4.10 is still functional then that's all you need for upgrading. Extract the \FBSLv3 folder from your RC2 download and copy it on top of your existing older \FBSLv3 installation folder wherever it is on your PC, and allow Windows to overwrite the older files whenever it asks you for a permission. Copy the other two folders from the zip anywhere at your option. They just contain a few examples of DynAsm and DynC unavailable in the earlier versions of FBSL. Read the included PDF for the description of new features.

Don't try to copy the \FBSLv3 folder anywhere else to "preserve" the older installation. It won't work. FBSL v3.5 is still a W.I.P. That's why it isn't fully functional without an earlier installation of FBSL and that's also why it doesn't yet have its own full-blown installer.

Most of your earlier sample scripts won't run in the new FBSL v3.5 environment but the PDF and the Newbie's board on the forum have instructions how to make the older scripts functional again. It's easy.

Don't regret you won't have access to the earlier v3.4.10 any more. It was much, much weaker than the new v3.5 and it is entirely deprecated and unsupported today.
Title: Re: Japi Experiments
Post by: Charles Pegge on December 11, 2016, 01:34:01 AM
Hi Roland,

Code: [Select]

NewTexture creates a new texture name and assigns a texture index number to it

  macro NewTexture(n)
  ===================
  static int n
  texe++
  n=texe
  end macro

DynSynthTexture generates a texture from a square block of pixels

Code: [Select]
  sub DynSynthTexture(int *n, any*v, int c)
  =========================================
  int w=sqr(c/4)
  MakeTexture @v, w, w, texn[n]
  end sub

PS:
inc/sysutil.inc has an uncommented datestamp (line 4). Sorry!   :o

Title: Re: Japi Experiments
Post by: Arnold on December 13, 2016, 02:33:01 AM
Hi Mike,

thanks for your info about FBSL 3.5. It will take some time until I will be able to create some meaningful code.

Roland