-
Notifications
You must be signed in to change notification settings - Fork 137
/
frag.glsl
160 lines (147 loc) · 3.87 KB
/
frag.glsl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
#version 400 compatibility
#extension GL_ARB_gpu_shader_fp64 : enable
#pragma optionNV(fastmath off)
#pragma optionNV(fastprecision off)
#define FLOAT float
#define VEC2 vec2
#define VEC3 vec3
#define AA_LEVEL 1
#define ESCAPE 1000.0
#define PI 3.141592653
#define FLAG_DRAW_MSET ((iFlags & 0x01) == 0x01)
#define FLAG_DRAW_JSET ((iFlags & 0x02) == 0x02)
#define FLAG_USE_COLOR ((iFlags & 0x04) == 0x04)
uniform vec2 iResolution;
uniform vec2 iCam;
uniform vec2 iJulia;
uniform float iZoom;
uniform int iType;
uniform int iIters;
uniform int iFlags;
uniform int iTime;
#define cx_one VEC2(1.0, 0.0)
VEC2 cx_mul(VEC2 a, VEC2 b) {
return VEC2(a.x*b.x - a.y*b.y, a.x*b.y + a.y*b.x);
}
VEC2 cx_sqr(VEC2 a) {
FLOAT x2 = a.x*a.x;
FLOAT y2 = a.y*a.y;
FLOAT xy = a.x*a.y;
return VEC2(x2 - y2, xy + xy);
}
VEC2 cx_cube(VEC2 a) {
FLOAT x2 = a.x*a.x;
FLOAT y2 = a.y*a.y;
FLOAT d = x2 - y2;
return VEC2(a.x*(d - y2 - y2), a.y*(x2 + x2 + d));
}
VEC2 cx_div(VEC2 a, VEC2 b) {
FLOAT denom = 1.0 / (b.x*b.x + b.y*b.y);
return VEC2(a.x*b.x + a.y*b.y, a.y*b.x - a.x*b.y) * denom;
}
VEC2 cx_sin(VEC2 a) {
return VEC2(sin(a.x) * cosh(a.y), cos(a.x) * sinh(a.y));
}
VEC2 cx_cos(VEC2 a) {
return VEC2(cos(a.x) * cosh(a.y), -sin(a.x) * sinh(a.y));
}
VEC2 cx_exp(VEC2 a) {
return exp(a.x) * VEC2(cos(a.y), sin(a.y));
}
//Fractal equations
VEC2 mandelbrot(VEC2 z, VEC2 c) {
return cx_sqr(z) + c;
}
VEC2 burning_ship(VEC2 z, VEC2 c) {
return VEC2(z.x*z.x - z.y*z.y, 2.0*abs(z.x * z.y)) + c;
}
VEC2 feather(VEC2 z, VEC2 c) {
return cx_div(cx_cube(z), cx_one + z*z) + c;
}
VEC2 sfx(VEC2 z, VEC2 c) {
return z * dot(z,z) - cx_mul(z, c*c);
}
VEC2 henon(VEC2 z, VEC2 c) {
return VEC2(1.0 - c.x*z.x*z.x + z.y, c.y * z.x);
}
VEC2 duffing(VEC2 z, VEC2 c) {
return VEC2(z.y, -c.y*z.x + c.x*z.y - z.y*z.y*z.y);
}
VEC2 ikeda(VEC2 z, VEC2 c) {
FLOAT t = 0.4 - 6.0/(1.0 + dot(z,z));
FLOAT st = sin(t);
FLOAT ct = cos(t);
return VEC2(1.0 + c.x*(z.x*ct - z.y*st), c.y*(z.x*st + z.y*ct));
}
VEC2 chirikov(VEC2 z, VEC2 c) {
z.y += c.y*sin(z.x);
z.x += c.x*z.y;
return z;
}
#if 1
#define DO_LOOP(name) \
for (i = 0; i < iIters; ++i) { \
VEC2 ppz = pz; \
pz = z; \
z = name(z, c); \
if (dot(z, z) > ESCAPE) { break; } \
sumz.x += dot(z - pz, pz - ppz); \
sumz.y += dot(z - pz, z - pz); \
sumz.z += dot(z - ppz, z - ppz); \
}
#else
#define DO_LOOP(name) \
for (i = 0; i < iIters; ++i) { \
z = name(z, c); \
if (dot(z, z) > ESCAPE) { break; } \
}
#endif
vec3 fractal(VEC2 z, VEC2 c) {
VEC2 pz = z;
VEC3 sumz = VEC3(0.0, 0.0, 0.0);
int i;
switch (iType) {
case 0: DO_LOOP(mandelbrot); break;
case 1: DO_LOOP(burning_ship); break;
case 2: DO_LOOP(feather); break;
case 3: DO_LOOP(sfx); break;
case 4: DO_LOOP(henon); break;
case 5: DO_LOOP(duffing); break;
case 6: DO_LOOP(ikeda); break;
case 7: DO_LOOP(chirikov); break;
}
if (i != iIters) {
float n1 = sin(float(i) * 0.1) * 0.5 + 0.5;
float n2 = cos(float(i) * 0.1) * 0.5 + 0.5;
return vec3(n1, n2, 1.0) * (1.0 - float(FLAG_USE_COLOR)*0.85);
} else if (FLAG_USE_COLOR) {
sumz = abs(sumz) / iIters;
vec3 n1 = sin(abs(sumz * 5.0)) * 0.45 + 0.5;
return n1;
} else {
return vec3(0.0, 0.0, 0.0);
}
}
float rand(float s) {
return fract(sin(s*12.9898) * 43758.5453);
}
void main() {
//Get normalized screen coordinate
vec2 screen_pos = gl_FragCoord.xy - (iResolution.xy * 0.5);
vec3 col = vec3(0.0, 0.0, 0.0);
for (int i = 0; i < AA_LEVEL; ++i) {
vec2 dxy = vec2(rand(i*0.54321 + iTime), rand(i*0.12345 + iTime));
VEC2 c = VEC2((screen_pos + dxy) * vec2(1.0, -1.0) / iZoom - iCam);
if (FLAG_DRAW_MSET) {
col += fractal(c, c);
}
if (FLAG_DRAW_JSET) {
col += fractal(c, iJulia);
}
}
col /= AA_LEVEL;
if (FLAG_DRAW_MSET && FLAG_DRAW_JSET) {
col *= 0.5;
}
gl_FragColor = vec4(clamp(col, 0.0, 1.0), 1.0 / (iTime + 1.0));
}